ES6篇(5)正则的扩展

26 篇文章 0 订阅


(1)RegExp 构造函数
在ES5中,RegExp 构造函数的参数有两种
1.var regex = new RegExp('xyz', 'i');
2.var regex = new RegExp(/xyz/i);
这两行代码都等价于,字面量方式var regex = /xyz/i;

但是ES5不允许下面这样写
var regex = new RegExp(/xyz/, 'i');会报错

Es6改变了这种行为,如果RegExp构造函数第一个参数是一个正则对象,那么可以使用第二个参数指定修饰符。而且,
返回的正则表达式会忽略原有的正则表达式的修饰符,只使用新指定的修饰符。
如:new RegExp(/abc/ig, 'i').flags
// "i",忽略了原来的ig

(2)字符串对象共有4个方法,可以使用正则表达式:match()、replace()、search()和split()。
ES6将这4个方法,在语言内部全部调用RegExp的实例方法,从而做到所有与正则相关的方法,全都定义在RegExp对象上。
String.prototype.match 调用 RegExp.prototype[Symbol.match]
String.prototype.replace 调用 RegExp.prototype[Symbol.replace]
String.prototype.search 调用 RegExp.prototype[Symbol.search]
String.prototype.split 调用 RegExp.prototype[Symbol.split]

(3)u修饰符
ES6 对正则表达式添加了u修饰符,含义为“Unicode 模式”,用来正确处理大于\uFFFF的 Unicode 字符。也就是说,
会正确处理四个字节的 UTF-16 编码。
/^\uD83D/u.test('\uD83D\uDC2A') // false
/^\uD83D/.test('\uD83D\uDC2A') // true  没有加u,将4个字节的UTF-16编码识别为两个字符,所以匹配成功了

加上u修饰符后,将改变一下正则表达式的行为
1.点字符
点(.)字符在正则表达式中,含义是除了换行符以外的任意单个字符。对于码点大于0xFFFF的 Unicode 字符,
点字符不能识别,加上u修饰符后可以识别了。
2.Unicode字符表示法
ES6 新增了使用大括号表示 Unicode 字符,这种表示法在正则表达式中必须加上u修饰符,才能识别当中的大括号,
否则会被解读为量词。
3.量词
使用u修饰符后,所有量词都会正确识别码点大于0xFFFF的 Unicode 字符。
例子:
/?{2}/.test('??') // false
/?{2}/u.test('??') // true

4.预定义模式
u修饰符也影响到预定义模式,能否正确识别码点大于0xFFFF的 Unicode 字符。
例子:
/^\S$/.test('?') // false
/^\S$/u.test('?') // true
上面代码的\S是预定义模式,匹配所有非空白字符。

5.i修饰符(忽略大小写)
有些 Unicode 字符的编码不同,但是字型很相近,比如,\u004B与\u212A都是大写的K。
例子:
/[a-z]/i.test('\u212A') // false
/[a-z]/iu.test('\u212A') // true
上面代码中,不加u修饰符,就无法识别非规范的K字符。

(4)正则实例对象新增unicode属性,表示是否设置了u修饰符
例子:
const r1 = /hello/;
const r2 = /hello/u;

r1.unicode // false
r2.unicode // true

(5)ES6新增y修饰符:粘连”(sticky)修饰符。
y修饰符的作用与g修饰符类似,也是全局匹配,后一次匹配都从上一次匹配成功的下一个位置开始。不同之处在于,g修饰符只要剩余位置中存在匹配就可,
而y修饰符确保匹配必须从剩余的第一个位置开始,这也就是“粘连”的涵义。
实际上,y修饰符号隐含了头部匹配的标志^。

/b/y.exec('aba')
// null
上面代码由于不能保证头部匹配,所以返回null。y修饰符的设计本意,就是让头部匹配的标志^在全局匹配中都有效。
(6)ES6 的正则实例对象多了sticky属性,表示是否设置了y修饰符。
例子:var r = /hello\d/y;
r.sticky // true

(7)ES6 为正则表达式新增了flags属性,会返回正则表达式的修饰符。
例子:/abc/ig.flags
// 'gi'

(8)s 修饰符:dotAll 模式
正则表达式中,点(.)是一个特殊字符,代表任意的单个字符,但是有两个例外。一个是四个字节的 UTF-16 字符,这个可以用u修饰符解决;
另一个是行终止符(line terminator character)。ES6引入s修饰符,使得.可以
匹配任意单个字符,这被称为dotAll模式。
例子:/foo.bar/s.test('foo\nbar') // true

正则表达式还引入了一个dotAll属性,返回一个布尔值,表示该正则表达式是否处在dotAll模式。
const re = /foo.bar/s;
// 另一种写法
// const re = new RegExp('foo.bar', 's');

re.test('foo\nbar') // true
re.dotAll // true
re.flags // 's'

注:终止符:
U+000A 换行符(\n)
U+000D 回车符(\r)
U+2028 行分隔符(line separator)
U+2029 段分隔符(paragraph separator)


(9)后行断言
JavaScript语言的正则表达式支持先行断言,和先行否定断言;不支持后行断言和后行否定断言
先行断言:x只有在y前面才匹配,必须写成/x(?=y)/
先行否定断言:x只有不在y前面才匹配,必须写成/x(?!y)/

后行断言:x只有在y后面才匹配,必须写成/(?<=y)x/
后行否定断言:x只有不在y后面才匹配,必须写成/(?<!y)x/

例子:
/(?<=\$)\d+/.exec('Benjamin Franklin is on the $100 bill')  // ["100"]
/(?<!\$)\d+/.exec('it’s is worth about €90')                // ["90"]

注意:“后行断言”的实现,需要先匹配/(?<=y)x/的x,然后再回到左边,匹配y的部分。这种“先右后左”的执行顺序,与所有其他正则操作相反,导致了一些不符合预期的行为。

(10)Unicode属性类
ES2018引入一种类的新写法:\p{...}和\P{...}

\p{...}:匹配满足条件的字符
\P{...}:匹配不满足条件的字符
Unicode属性类要指定属性名和属性值,对于某些属性可以只写属性名,或只写属性值
注意:这两种类只对 Unicode 有效,所以使用的时候一定要加上u修饰符。如果不加u修饰符,正则表达式使用\p和\P会报错,ECMAScript 预留了这两个类。
例子:
// 匹配所有空格
\p{White_Space}
// 匹配所有的箭头字符
const regexArrows = /^\p{Block=Arrows}+$/u;
regexArrows.test('←↑→↓↔↕↖↗↘↙⇏⇐⇑⇒⇓⇔⇕⇖⇗⇘⇙⇧⇩') // true

由于 Unicode 的各种属性非常多,所以这种新的类的表达能力非常强。


(11)具名组匹配
正则表达式使用圆括号进行组匹配。
例子:
const RE_DATE = /(\d{4})-(\d{2})-(\d{2})/;

const matchObj = RE_DATE.exec('1999-12-31');
const year = matchObj[1]; // 1999
const month = matchObj[2]; // 12
const day = matchObj[3]; // 31
使用exec方法,可以将三组匹配的结果提取出来,组匹配的缺点在于,每一组
匹配的含义不容易看出,且组的顺序如果变了,引用时需要修改序号。

ES2018引入了具名组匹配,允许为每一组匹配指定名字,易于阅读和引用。
例子:
const RE_DATE = /(?<year>\d{4})-(?<month>\d{2})-(?<day>\d{2})/;

const matchObj = RE_DATE.exec('1999-12-31');
const year = matchObj.groups.year; // 1999
const month = matchObj.groups.month; // 12
const day = matchObj.groups.day; // 31

具名组 ?<year>
引用matchObj.groups.year

如果具名组没有匹配,那么对应的groups对象的属性是undefined。

具名组的应用:解构赋值和替换
例子1:
let {groups: {one, two}} = /^(?<one>.*):(?<two>.*)$/u.exec('foo:bar');
one  // foo
two  // bar

例子2:let re = /(?<year>\d{4})-(?<month>\d{2})-(?<day>\d{2})/u;

'2015-01-02'.replace(re, '$<day>/$<month>/$<year>')

引用:如果要在正则表达式内部引用某个“具名组匹配”,可以使用\k<组名>的写法。

const RE_TWICE = /^(?<word>[a-z]+)!\k<word>$/;
RE_TWICE.test('abc!abc') // true
RE_TWICE.test('abc!ab') // false
数字引用(\1)依然有效。

const RE_TWICE = /^(?<word>[a-z]+)!\1$/;
RE_TWICE.test('abc!abc') // true
RE_TWICE.test('abc!ab') // false
这两种引用语法还可以同时使用。

const RE_TWICE = /^(?<word>[a-z]+)!\k<word>!\1$/;
RE_TWICE.test('abc!abc!abc') // true
RE_TWICE.test('abc!abc!ab') // false

(12)String.prototype.matchAll
可以一次性取出所有匹配。不过,它返回的是一个遍历器(Iterator),而不是数组。

阮一峰 ES6 正则扩展

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值