ES6之正则的扩展
RegExp构造函数
在ES5中我们可以这样声明一个正则表达式:
1 | var regex = new RegExp('xyz', 'i'); |
但是这样会报错,因为第一个参数不是字符串,而是一个正则表达式:
1 | var regex = new RegExp(/xyz/, 'i'); |
但是在ES6中支持这种行为,并且如果存在第二个参数,会忽略原有的修饰符:
1 | var reg = new RegExp(/abc/ig, 'i'); |
字符串的正则方法
match
match 方法会返回一个数组,它包括整个匹配结果,和通过捕获组匹配到的结果,如果没有匹配到则返回null
这个方法在 String.prototype.match() 的内部调用。例如,下面的两个方法返回相同结果。
1 | 'abc'.match(/a/); |
replace
返回用替换器替换相应匹配项后的新字符串。
1 | 'abc'.replace(/a/, 'A'); |
search
如果成功的话,[@@search]() 返回该正则模式的第一个匹配项的在字符串中的位置索引。否则将返回-1。
1 | 'abc'.search(/a/); |
split
切割字符串
返回包含其子字符串的数组。
1 | 'a-b-c'.split(/-/); // ["a", "b", "c"] |
修饰符
u修饰符
ES6 对正则表达式添加了u修饰符,含义为“Unicode 模式”,用来正确处理大于\uFFFF的 Unicode 字符。
1 | /^\uD83D/u.test('\uD83D\uDC2A') // false |
ES5 不支持四个字节的 UTF-16 编码,会将其识别为两个字符,导致第二行代码结果为true。
RegExp.prototype.unicode
查看正则对象是否设置了u修饰符。
1 | const r1 = /hello/; |
i修饰符
表示不区分大小写
并且和u修饰符一起用,可以识别非规范的字符
1 | /[a-z]/i.test('\u212A') // false |
y修饰符
y修饰符也叫做粘连修饰符
1 | var s = 'aaa_aa_a'; |
使用y修饰符每次匹配,都是从剩余字符串的头部开始。所以上面代码的第二次匹配的头部是_,故匹配不到。
y修饰符的设计本意,就是让头部匹配的标志^在全局匹配中都有效。
RegExp.prototype.sticky
与y修饰符相匹配,ES6 的正则实例对象多了sticky属性,表示是否设置了y修饰符。
1 | var r = /hello\d/y; |
RegExp.prototype.flags
1 | // ES5 的 source 属性 |
s修饰符
ES2018 引入s修饰符,使得.可以匹配任意单个字符。
正则表达式中,点(.)是一个特殊字符,代表任意的单个字符,但是有两个例外。一个是四个字节的 UTF-16 字符,这个可以用u修饰符解决;另一个是行终止符(line terminator character)。
1 | /foo.bar/s.test('foo\nbar') // true |
后行断言
ES2018引入后行断言
什么是先行断言
“先行断言”指的是,x只有在y前面才匹配,必须写成/x(?=y)/。比如,只匹配百分号之前的数字,要写成/\d+(?=%)/。“先行否定断言”指的是,x只有不在y前面才匹配,必须写成/x(?!y)/。比如,只匹配不在百分号之前的数字,要写成/\d+(?!%)/。
1 | /\d+(?=%)/.exec('100% of US presidents have been male') // ["100"] |
上面两个字符串,如果互换正则表达式,就不会得到相同结果。另外,还可以看到,“先行断言”括号之中的部分((?=%)),是不计入返回结果的。
什么是后行断言
“后行断言”正好与“先行断言”相反,x只有在y后面才匹配,必须写成/(?<=y)x/。比如,只匹配美元符号之后的数字,要写成/(?<=\$)\d+/。“后行否定断言”则与“先行否定断言”相反,x只有不在y后面才匹配,必须写成/(?<!y)x/。比如,只匹配不在美元符号后面的数字,要写成/(?<!\$)\d+/。
1 | /(?<=\$)\d+/.exec('Benjamin Franklin is on the $100 bill') // ["100"] |
上面的例子中,“后行断言”的括号之中的部分((?<=\$)),也是不计入返回结果。
具体名匹配
ES2018 引入了具名组匹配(Named Capture Groups),允许为每一个组匹配指定一个名字,既便于阅读代码,又便于引用。
不使用具体组匹配的写法:
1 | const RE_DATE = /(\d{4})-(\d{2})-(\d{2})/; |
matchObj的值是一个类数组对象:
1 | [ '1999-12-31', |
使用具体组匹配的写法:
1 | const RE_DATE = /(?<year>\d{4})-(?<month>\d{2})-(?<day>\d{2})/; |
matchObj的值是一个类数组对象,但是groups属性有值:
1 | [ '1999-12-31', |
解构赋值和替换
有了具名组匹配以后,可以使用解构赋值直接从匹配结果上为变量赋值。
1 | let {groups: {one, two}} = /^(?<one>.*):(?<two>.*)$/u.exec('foo:bar'); |
引用
如果要在正则表达式内部引用某个“具名组匹配”,可以使用\k<组名>的写法。
1 | const RE_TWICE = /^(?<word>[a-z]+)!\k<word>$/; |
数字引用(\1)依然有效。
\1是反向引用的意思,表示获取第一个()匹配的引用。
1 | const RE_TWICE = /^(?<word>[a-z]+)!\1$/; |
当然这两种引用语法还可以同时使用:
1 | const RE_TWICE = /^(?<word>[a-z]+)!\k<word>!\1$/; |
ps: 正则的用法妙不可言,平时用的比较少,就记一下这些常用的吧。