下面的正则表达式能够匹配kidkidkid
:正则表达式
/kidkidkid/
而另外一种更优雅的写法是:函数
/(kid){3}/
这里由圆括号包裹的一个小总体称为分组。prototype
一个分组中,能够有多个候选表达式,用|
分隔:code
var reg = /I love (him|her|it)/; reg.test('I love him') // true reg.test('I love her') // true reg.test('I love it') // true reg.test('I love them') // false
这里的|
至关于“或”的意思。字符串
被正则表达式匹配(捕获)到的字符串会被暂存起来。其中,由分组捕获的串会从1开始编号,因而咱们能够引用这些串:get
var reg = /(\d{4})-(\d{2})-(\d{2})/ var date = '2010-04-12' reg.test(date) RegExp.$1 // 2010 RegExp.$2 // 04 RegExp.$3 // 12
$1
引用了第一个被捕获的串,$2
是第二个,依次类推。博客
replace
配合String.prototype.replace
方法的传参中能够直接引用被捕获的串。好比咱们想将日期12.21/2012
改成2012-12-21
:it
var reg = /(\d{2}).(\d{2})\/(\d{4})/ var date = '12.21/2012' date = date.replace(reg, '$3-$1-$2') // date = 2012-12-21
顺道一提,给replace
传迭代函数,有时能优雅地解决一些问题。io
将违禁词转换为等字数的星号是一个常见功能。好比文本是kid is a doubi
,其中kid
与doubi
是违禁词,那么转换后应该为*** is a *****
。咱们能够这么写:function
var reg = /(kid|doubi)/g var str = 'kid is a doubi' str = str.replace(reg, function(word){ return word.replace(/./g, '*') })
若是碰到相似/((kid) is (a (doubi)))/
的嵌套分组,捕获的顺序是什么?来试试:
var reg = /((kid) is (a (doubi)))/ var str = "kid is a doubi" reg.test( str ) // true RegExp.$1 // kid is a doubi RegExp.$2 // kid RegExp.$3 // a doubi RegExp.$4 // doubi
规则是以左括号出现的顺序进行捕获。
正则表达式里也能进行引用,这称为反向引用:
var reg = /(\w{3}) is \1/ reg.test('kid is kid') // true reg.test('dik is dik') // true reg.test('kid is dik') // false reg.test('dik is kid') // false
\1
引用了第一个被分组所捕获的串,换言之,表达式是动态决定的。
注意,若是编号越界了,则会被当成普通的表达式:
var reg = /(\w{3}) is \6/; reg.test( 'kid is kid' ); // false reg.test( 'kid is \6' ); // true
分组有四种类型:
捕获型 - ()
非捕获型 - (?:)
正向前瞻型 - (?=)
反向前瞻型 - (?!)
咱们以前说的都是捕获型分组,只有这种分组会暂存匹配到的串。
有时候,咱们只是想分个组,而没有捕获的需求,则可使用非捕获型分组,语法为左括号后紧跟?:
:
var reg = /(?:\d{4})-(\d{2})-(\d{2})/ var date = '2012-12-21' reg.test(date) RegExp.$1 // 12 RegExp.$2 // 21
这个例子中,(?:\d{4})
分组不会捕获任何串,因此$1
为(\d{2})
捕获的串。
就好像你站在原地,向前眺望:
正向前瞻型分组 - 你前方是什么东西吗?
负向前瞻型分组 - 你前方不是什么东西吗?
太拗口了,我喜欢称之为确定表达式与否认表达式。先举个正向前瞻的例子:
var reg = /kid is a (?=doubi)/ reg.test('kid is a doubi') // true reg.test('kid is a shabi') // false
kid is a
后面跟着什么?若是是doubi
才能匹配成功。
而负向前瞻则恰好相反:
var reg = /kid is a (?!doubi)/ reg.test('kid is a doubi') // false reg.test('kid is a shabi') // true
若是前瞻型分组也不会捕获值。那么它与非捕获型的区别是什么?看例子:
var reg, str = "kid is a doubi" reg = /(kid is a (?:doubi))/ reg.test(str) RegExp.$1 // kid is a doubi reg = /(kid is a (?=doubi))/ reg.test(str) RegExp.$1 // kis is a
可见,非捕获型分组匹配到的串,仍会被外层的捕获型分组捕获到,但前瞻型却不会。当你须要参考后面的值,又不想连它一块儿捕获时,前瞻型分组就派上用场了。
最后,JS不支持后瞻型分组。
原创,自由转载,请署名,本人博客 kid-wumeng.me谢谢