首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >IPv6正则表达式(RegEx)不能在PL/SQL中工作

IPv6正则表达式(RegEx)不能在PL/SQL中工作
EN

Stack Overflow用户
提问于 2014-02-21 17:12:53
回答 3查看 1.7K关注 0票数 2

我似乎找不出为什么这个正则表达式不能在PL/SQL中工作。

代码语言:javascript
复制
if ( REGEXP_LIKE(v,'/^(?>(?>([a-f0-9]{1,4})(?>:(?1)){7}|(?!(?:.*[a-f0-9](?>:|$)){8,})((?1)(?>:(?1)){0,6})?::(?2)?)|(?>(?>(?1)(?>:(?1)){5}:|(?!(?:.*[a-f0-9]:){6,})(?3)?::(?>((?1)(?>:(?1)){0,4}):)?)?(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])(?>\.(?4)){3}))$/iD') ) then

它用于验证IPv4和IPv6,它来自于:https://stackoverflow.com/a/1934546/3112803

不确定这是否与此有关,但我也问了关于末尾的D标志的问题:What Does This Regular Expression (RegEx) Flag Mean /iD

出于某种原因,这个正则表达式适用于,我在这个站点上测试的大多数是:http://regex101.com/,但是在PL/SQL中,一切都是无效的。

我所说的most的意思是,在某些情况下,我发现它失败了,但是我已经搜索了好几天,这是我能找到的最好的一个,在512个字符以下(512个是在PL/SQL中使用REGEXP_LIKE的限制)。

我很感谢你的帮助。谢谢!

这些是我用的测试用例..。

代码语言:javascript
复制
{1: Initial address, regex should say valid/match}
select isValid('2001:0db8:0000:0000:0000:ff00:0042:8329','ipv6') from dual;

{2: After removing all leading zeroes, regex should say valid/match}
select isValid('2001:db8:0:0:0:ff00:42:8329','ipv6') from dual;

{3: After omitting consecutive sections of zeroes, regex should say valid/match}
select isValid('2001:db8::ff00:42:8329','ipv6') from dual;

{4: The loopback address, regex should say valid/match}
select isValid('0000:0000:0000:0000:0000:0000:0000:0001','ipv6') from dual;

{5: The loopback address be abbreviated to ::1 by using both rules, regex should say valid/match}
select isValid('::1','ipv6') from dual;

{6: This should be valid/match}
select isValid('ABCD:ABCD:ABCD:ABCD:ABCD:ABCD:192.168.158.190','ipv6') from dual;

{7: This should be valid/match}
select isValid('::','ipv6') from dual;

{8: IPv6 applications to communicate directly with IPv4 applications, regex should say valid/match}
select isValid('0:0:0:0:0:ffff:192.1.56.10','ipv6') from dual;

{9: should NOT be valid/match}
select isValid('::ffff:192.1.56.10/96','ipv6') from dual;

{old formats used for tunneling, these should NOT be valid/matches}
{10}
select isValid('0:0:0:0:0:0:192.1.56.10','ipv6') from dual;
{11}
select isValid('::192.1.56.10/96','ipv6') from dual;

{These 4 should be valid/match}
{12}
select isValid('::FFFF:129.144.52.38','ipv6') from dual;
{13}
select isValid('::129.144.52.38','ipv6') from dual;
{14}
select isValid('::FFFF:d','ipv6') from dual;
{15}
select isValid('1080:0:0:0:8:800:200C:417A','ipv6') from dual;

{These 4 should NOT be valid/match}
{16}
select isValid('::FFFF:d.d.d','ipv6') from dual;
{17}
select isValid('::FFFF:d.d','ipv6') from dual;
{18}
select isValid('::d.d.d','ipv6') from dual;
{19}
select isValid('::d.d','ipv6') from dual;

我被告知测试#6是错误的,ABCD:ABCD:ABCD:ABCD:ABCD:ABCD:192.168.158.190不是一个有效的IPv6地址,对吗?

测试用例8-11来自这里:http://publib.boulder.ibm.com/infocenter/iseries/v5r3/index.jsp?topic=%2Frzai2%2Frzai2ipv6addrformat.htm,但我被告知10和11不再使用。

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2014-02-21 19:53:33

与其在单个正则表达式中执行所有操作,不如将正则表达式分解为较小的正则表达式并进行测试:

代码语言:javascript
复制
if (
    /* IPv6 expanded */
    REGEX_LIKE(v, '\A[a-f0-9]{1,4}(:[a-f0-9]{1,4}){7}\z', 'i')
    /* IPv6 shorthand */
    OR (NOT REGEX_LIKE(v, '\A(.*?[a-f0-9](:|\z)){8}', 'i')
        AND REGEX_LIKE(v, '\A([a-f0-9]{1,4}(:[a-f0-9]{1,4}){0,6})?::([a-f0-9]{1,4}(:[a-f0-9]{1,4}){0,6})?\z', 'i'))
    /* IPv6 dotted-quad notation, expanded */
    OR REGEX_LIKE(v, '\A[a-f0-9]{1,4}(:[a-f0-9]{1,4}){5}:(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])(\.(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])){3}\z', 'i')
    /* IPv6 dotted-quad notation, shorthand */
    OR (NOT REGEX_LIKE(v, '\A(.*?[a-f0-9]:){6}', 'i')
        AND REGEX_LIKE(v, '\A([a-f0-9]{1,4}(:[a-f0-9]{1,4}){0,4})?::([a-f0-9]{1,4}:){0,5}(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])(\.(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])){3}\z', 'i'))
   ) then

这只是对IPv6的测试。不允许使用IPv4。

由于PL/SQL风味没有子例程调用(?n),因此别无选择,只能扩展所有内容。而缺乏负前瞻性的(?!pattern)迫使我们用两个regex测试操作来模拟它。

\A\z用于匹配字符串的开始和结束,因为它们都不受标志的影响,\z行为与D模式下的$行为相同。

票数 5
EN

Stack Overflow用户

发布于 2014-02-21 17:43:19

您必须从一开始就去掉/和/iD,这是perl语法的一部分,表明它是正则表达式。

最后的i开关意味着忽略大小写,可以作为regexp_like的额外参数给出,因此:

代码语言:javascript
复制
if ( REGEXP_LIKE(v,'^(?>(?>([a-f0-9]{1,4})(?>:(?1)){7}|(?!(?:.*[a-f0-9](?>:|$)){8,})((?1)(?>:(?1)){0,6})?::(?2)?)|(?>(?>(?1)(?>:(?1)){5}:|(?!(?:.*[a-f0-9]:){6,})(?3)?::(?>((?1)(?>:(?1)){0,4}):)?)?(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])(?>\.(?4)){3}))$','i') ) the

还有更多的问题,因为perl正则表达式不是100%等效于oracle正则表达式,我看到这里使用的模式没有像?>那样可用,也许您可以将regexp拆分到ipv4和ipv6之间,以避免达到oracle中的限制。只需做REGEXP_LIKE(ip,'ipv4pattern')或REGEXP_LIKE(ip,'ipv6pattern')

将上述正则表达式的ipv4部分调整为在甲骨文中工作的部分,我将得到以下信息:

代码语言:javascript
复制
REGEXP_LIKE(ip,'^((25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])\.){3}(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])$','i')
票数 3
EN

Stack Overflow用户

发布于 2014-03-14 19:12:11

代码语言:javascript
复制
REGEXP_LIKE(ip,'^(([\dA-F]{1,4}:([\dA-F]{1,4}:([\dA-F]{1,4}:([\dA-F]{1,4}:([\dA-F]{1,4}:[\dA-F]{0,4}|:[\dA-F]{1,4})?|(:[\dA-F]{1,4}){0,2})|(:[\dA-F]{1,4}){0,3})|(:[\dA-F]{1,4}){0,4})|:(:[\dA-F]{1,4}){0,5})((:[\dA-F]{1,4}){2}|:(25[0-5]|(2[0-4]|1\d|[1-9])?\d)(\.(25[0-5]|(2[0-4]|1\d|[1-9])?\d)){3})|(([\dA-F]{1,4}:){1,6}|:):[\dA-F]{0,4}|([\dA-F]{1,4}:){7}:)\z', 'i')

http://home.deds.nl/~aeron/regex/的XML regex中修改

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/21940807

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档