javascript正则表达式中 (?=exp)、(?<=exp)、(?!exp)
(?=exp)
百度百科给的解释:非获取匹配,正向肯定预查,在任何匹配pattern的字符串开始处匹配查找字符串,该匹配不需要获取供以后使用。例如,“Windows(?=95|98|NT|2000)”能匹配“Windows2000”中的“Windows”,但不能匹配“Windows3.1”中的“Windows”。预查不消耗字符,也就是说,在一个匹配发生后,在最后一次匹配之后立即开始下一次匹配的搜索,而不是从包含预查的字符之后开始。
只看这个,看一辈子恐怕也看不明白。
我们看个案例:
console.log(('123456789.56').match(/\d{1,3}(?=(\d{3})+(\.\d*)?$)/g)); //结果: ["123", "456"]
解释:
首先要满足 : \d{1,3}这个正则。
第 1 步: 开始的 1 满足\d{1,3}最大匹配,后面的 23456789.56 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “1” 不满足。
第 2 步:开始的 12 满足 \d{1,3}最大匹配,后面的 3456789.56 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “12” 不满足。
第 3 步:开始的 123 满足 \d{1,3}最大匹配,后面的 456789.56 满足 (?=(\d{3})+(\.\d*)?$),所以这个 “123” 满足,返回 “123”
第 4 步: 从4开始搜索,开始的 4 满足\d{1,3}最大匹配,后面的 56789.56 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “4 ”不满足。
第 5 步: 从5开始搜索,开始的 45 满足\d{1,3}最大匹配,后面的 6789.56 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “45“ 不满足。
第 6 步: 从6开始搜索,开始的 456 满足\d{1,3}最大匹配,后面的 789.56 满足 (?=(\d{3})+(\.\d*)?$),所以这个 “456“ 满足,返回 “456”
第 7 步: 从7开始搜索,开始的 7 满足\d{1,3}最大匹配,后面的 89.56 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “7“ 不满足。
第 8 步: 从8开始搜索,开始的 78 满足\d{1,3}最大匹配,后面的 9.56 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “78“ 不满足。
第 9 步: 从9开始搜索,开始的 789 满足\d{1,3}最大匹配,后面的 .56 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “789“ 不满足,此时“789”满足了 \d{1,3}最大的匹配,但是后面没有能匹配的。
第 10 步: 从 . 开始搜索,开始的 .(.|9. | 89. )都不满足\d{1,3},所以这个 从 . 位置搜索的没满足的。
第 11 步 : 从 5 开始搜索,开始的 5 满足 \d{1,3}最大匹配,后面的6 不满足 (?=(\d{3})+(\.\d*)?$),所以这个 “5“ 不满足。
第 12 步 : 从 6 开始搜索,开始的 56 满足 \d{1,3}最大匹配,后面就没有了,不满足 (?=(\d{3})+(\.\d*)?$), 所以这个 “56 “ 不满足。
说的直接点就是: (?=exp) 里面匹配的内容是不会被 match捕获的,只是当个条件来对待使用,只是用来判断当前检索的位置后面的能否满足这个条件。
(?<=exp)
也叫零宽度正回顾后发断言,它断言自身出现的位置的前面能匹配表达式exp。比如(?<=\bre)\w+\b会匹配以re开头的单词的后半部分(除了re以外的部分),例如在查找reading a book时,它匹配ading。
有点奇怪,在js 的正则表达式中来测试这个一直报错,猜测估计是不支持。
(?!exp)
这个与 (?=exp) 相反.
console.log(('123456789.56').match(/\d{1,3}(?!(\d{3})+(\.\d*)?$)/g));//["12", "345", "678", "9", "56"]
解释:
首先的满足 \d{1,3}
从 1 处开始搜索,\d{1,3}的最大匹配是 1,后面是 23456789.56 ,满足, (?!(d{3}) +(\.\d*)?$), OK, 此时 \d{1,3} 匹配的是 “1”
第二次从 2 开始搜索,,\d{1,3}的最大匹配是 12, 后面是 3456789.56 ,满足, (?!(d{3}) +(\.\d*)?$), OK, 此时 \d{1,3} 匹配的是 “12”
第三次从 3 开始搜索,\d{1,3}的最大匹配是 123,后面是 456789.56 ,不满足, (?!(d{3}) +(\.\d*)?$),所以返回的结果数组中第一个是 “12”., 此时 \d{1,3} 匹配的最大是是 “12”
第四次从 4 开始搜索, 后面是 56789.56 ,\d{1,3}的最大匹配是 34,满足, (?!(d{3}) +(\.\d*)?$), OK, 此时 \d{1,3} 匹配的是 “34”
第五次从 5 开始搜索, 后面是 6789.56 ,\d{1,3}的最大匹配是 345,满足, (?!(d{3}) +(\.\d*)?$), OK, 此时 \d{1,3} 匹配的是 “345”.,已经满足最大3个数字了,此时返回 “345”
第六次从 6 开始搜索, 后面是 789.56 ,\d{1,3}的最大匹配是 6,不满足, (?!(d{3}) +(\.\d*)?$), 此时 \d{1,3} 匹配的是 “6”.
第七次从 7 开始搜索, 后面是 89.56 ,\d{1,3}的最大匹配是 67,满足, (?!(d{3}) +(\.\d*)?$), 此时 \d{1,3} 匹配的是 “67”.
第八次从 8 开始搜索, 后面是 9.56 ,\d{1,3}的最大匹配是 678,满足, (?!(d{3}) +(\.\d*)?$), 此时 \d{1,3} 匹配的是 “678”.,已经满足最大3个数字了,此时返回 “678”.
第九次从 9 开始搜索, 后面是 .56 ,\d{1,3}的最大匹配是 9,满足, (?!(d{3}) +(\.\d*)?$), 此时 \d{1,3} 匹配的是 “9”
第十次从 . 开始搜索, 此时 “9.” 不满足 \d{1,3} ,所以此时返回 “9”
第十一次从 5 开始搜索, 后面是 6 ,\d{1,3}的最大匹配是 6,满足, (?!(d{3}) +(\.\d*)?$), 此时 \d{1,3} 最大匹配的是 “5”
第十二次从 6 开始搜索, 后面没有 ,\d{1,3}的最大匹配是 56,满足, (?!(d{3}) +(\.\d*)?$), 此时 \d{1,3} 最大匹配的是 “56”,结束,返回最后一个 “56”
这只是个人的理解,有不对之处,望各位大神们指正。
javascript正则表达式中 (?=exp)、(?<=exp)、(?!exp)的更多相关文章
- JavaScript 正则表达式中的特殊字符
正则表达式中的特殊字符 字符 含义 \ 依照下列规则匹配: 在非特殊字符之前的反斜杠表示下一个字符是特殊字符,不能按照字面理解.例如,前面没有 "\" 的 "b" ...
- 【转载】【网络安全】渗透中 PoC、Exp、Payload 与 Shellcode 的区别
原文地址 渗透中 PoC.Exp.Payload 与 Shellcode 的区别 概念 PoC,全称"Proof of Concept",中文"概念验证",常指 ...
- JavaScript正则表达式详解(二)JavaScript中正则表达式函数详解
二.JavaScript中正则表达式函数详解(exec, test, match, replace, search, split) 1.使用正则表达式的方法去匹配查找字符串 1.1. exec方法详解 ...
- javascript中正则表达式中的 match,exec,test,replace 之我理解
这个正则 ($&) 的语法: https://msdn.microsoft.com/library/3k9c4a32(v=vs.94).aspx 在ECMAScript中对这几个的说明: ma ...
- JavaScript —— 字符串中使用正则表达式
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 我所认识的javascript正则表达式
前言 如果说这是一篇关于正则表达式的小结,我更愿意把它当做一个手册. 目录:(点击可直达) RegExp 三大方法(test.exec.compile) String 四大护法(search.matc ...
- javascript正则表达式(一)
元字符 ( [ { \ ^ $ | ) ? * + . 预定义的特殊字符 字符 正则 描述 \t /\t/ 制表符 \n /\n/ 制表符 \r /\r/ 回车符 \f /\f/ 换页符 \a /\a ...
- DOM笔记(十):JavaScript正则表达式
一.RegExp ECMAScript通过RegExp类型类支持正则表达式,语法和Perl类似: var exp = /pattern/flags; patternb部分是任何简单的或复杂的正则表达式 ...
- JavaScript正则表达式基础
ECMAScript 3 开始支持正则表达式,其语法和 Perl 语法很类似,一个完整的正则表达式结构如下: var expression = / pattern / flags ; 其中,模式(pa ...
随机推荐
- 位运算 - a^b
求 a 的 b 次方对 p 取模的值. 输入格式 三个整数 a,b,p ,在同一行用空格隔开. 输出格式 输出一个整数,表示a^b mod p的值. 数据范围 1≤a,b,p≤109 输入样例: 3 ...
- Java类加载过程及static详解
类从被加载到JVM中开始,到卸载为止,整个生命周期包括:加载.验证.准备.解析.初始化.使用和卸载七个阶段. 其中类加载过程包括加载.验证.准备.解析和初始化五个阶段. 类加载器的任务就是根据一个类的 ...
- svn安装时遇到问题总结
问题1: 一番折腾终于解决了,现将解决方法总结一下: 1.点击Window键+R键,如下图: 2.输入services.msc命令,然后点击“确定”,得到下图: 3.找到并选中“Windows Man ...
- redis.conf 配置信息:读取及修改命令
相关资源 网址 官方地址(网页中 Command + F,输入井号"#",方便查看没有注释的行) http://download.redis.io/redis-stable/red ...
- webpack 中,module、chunk、bundle 的区别(待补充)
项目 区别 module 是开发中的单个模块 chunk 中文意思是"块",是指 webpack 在进行模块依赖分析的时候,代码分割出来的代码块 bundle
- win10系统IE浏览器中无法显示Java国际化的问题
http://jingyan.baidu.com/article/656db918e37914e381249c9f.html?qq-pf-to=pcqq.c2c 自从装上Win10系统后,跟随前些IE ...
- 黄聪:微信URL Scheme,URL唤起微信
微信URL Scheme 在外部浏览器中,可以通过<a href="weixin://">打开微信APP 也可以通过加一些参数,打开微信APP里的指定页面 <a ...
- 在Docker中监控Java应用程序的5个方法
译者注:Docker是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的Linux机器上,也可以实现虚拟化.通常情况下,监控的主要目的在于:减少宕机 ...
- linux查看用户登录,操作历史等
who 命令:显示当前当登录的用户的信息 who -b命令:显示系统最近一次的启动时间 w 命令:显示登录的用户及其当前执行的任务 last 命令:显示当前与过去登录系统的用户的信息 lastb 命令 ...
- [UE4]Menu Anchor,菜单锚点
一.想要弹出某个菜单的时候,Menu Anchor可以做为菜单弹出的位置. 二.Menu Anchor本身不显示任何东西 三.Menu Class:选择要弹出的UI,可以是任意的UserWidget ...