网上正则表达式的教程够多了,但因为javascript的历史比较悠久,也比较古老,所以有许多特性是不支持的。咱们先从最简单地提及,文章所演示的正则基本都是perl方式。javascript
元字符
( [ { \ ^ $ | ) ? * + .html
预约义的特殊字符
字符 |
正则 |
描述 |
\t |
/\t/ |
制表符 |
\n |
/\n/ |
制表符 |
\r |
/\r/ |
回车符 |
\f |
/\f/ |
换页符 |
\a |
/\a/ |
alert字符 |
\e |
/\e/ |
escape字符 |
\cX |
/\cX/ |
与X相对应的控制字符 |
\b |
/\b/ |
与回退字符 |
\v |
/\v/ |
垂直制表符 |
\0 |
/\0/ |
空字符 |
字符类
简单类前端
原则上正则的一个字符对应一个字符,咱们能够用[]把它们括起来,让[]这个总体对应一个字符。如java
alert(/ruby/.test( "ruby" )); //true |
alert(/[abc]/.test( "a" )); //true |
alert(/[abc]/.test( "b" )); //true |
alert(/[abc]/.test( "c" )); //true |
alert( "a bat ,a Cat,a fAt bat ,a faT cat" .match(/[bcf]at/gi)); //bat,Cat,fAt,bat,faT,cat |
负向类正则表达式
也是在那个括号里作文章,前面加个元字符进行取反,表示匹配不能为括号里面的字符。api
alert(/[^abc]/.test( "a" )); //false |
alert(/[^abc]/.test( "b" )); //false |
alert(/[^abc]/.test( "6" )); //true |
alert(/[^abc]/.test( "gg" )); //true |
范围类浏览器
仍是在那个中括号里面作文章。有时匹配的东西过多,并且类型又相同,所有输入太麻烦,咱们能够用它。特征就是在中间加了个横线。ruby
组合类this
仍是在那个中括号里面作文章。容许用中括号匹配不一样类型的单个字符。spa
alert(/[a-f]/.test( "b" )); //true |
alert(/[a-f]/.test( "k" )); //false |
alert(/[a-z]/.test( "h" )); //true |
alert(/[A-Z]/.test( "gg" )); //false |
alert(/[^H-Y]/.test( "G" )); //true |
alert(/[0-9]/.test( "8" )); //true |
alert(/[^7-9]/.test( "6" )); //true |
alert(/[a-m1-5\n]/.test( "a" )) //true |
alert(/[a-m1-5\n]/.test( "3" )) //true |
alert(/[a-m1-5\n]/.test(a)) //true |
alert(/[a-m1-5\n]/.test( "r" )) //false |
预约义类
仍是在那个中括号里面作文章,不过它好像已经走到尽头了。因为是中括号的马甲,所以它们仍是对应一个字符。
字符 |
等同于 |
描述 |
. |
[^\n\r] |
除了换行和回车以外的任意字符 |
\d |
[0-9] |
数字字符 |
\D |
[^0-9] |
非数字字符 |
\s |
[ \t\n\x0B\f\r] |
空白字符 |
\S |
[^ \t\n\x0B\f\r] |
非空白字符 |
\w |
[a-zA-Z_0-9] |
单词字符(全部的字母) |
\W |
[^a-zA-Z_0-9] |
非单词字符 |
alert(/\d/.test( "3" )) //true |
alert(/\d/.test( "w" )) //false |
alert(/\D/.test( "w" )) //true |
alert(/\w/.test( "w" )) //true |
alert(/\w/.test( "司" )) //false |
alert(/\W/.test( "徒" )) //true |
alert(/\s/.test( " " )) //true |
alert(/\S/.test( " " )) //false |
alert(/\S/.test( "正" )) //true |
alert(/./.test( "美" )) //true |
alert(/./.test( " " )) //true |
量词
因为元字符与特殊字符或字符类或者它们的组合(中括号)甚至它们的马甲(预约义类)都是一对一进行匹配。咱们要匹配“司徒正美这个词”,最简单都要/..../,若是长到50多个字符岂不是要死人。所以咱们逼切须要一个简单的操做,来处理这数量关系。
简单量词
代码 |
类型 |
描述 |
? |
软性量词 |
出现零次或一次 |
* |
软性量词 |
出现零次或屡次(任意次) |
+ |
软性量词 |
出现一次或屡次(至道一次) |
{n} |
硬性量词 |
对应零次或者n次 |
{n,m} |
软性量词 |
至少出现n次但不超过m次 |
{n,} |
软性量词 |
至少出现n次(+的升级版) |
alert(/..../.test( "司徒正美" )) //true |
alert(/司徒正美/.test( "司徒正美" )) //true |
alert(/[\u4e00-\u9fa5]{4}/.test( "司徒正美" )) //true |
alert(/[\u4e00-\u9fa5]{4}/.test( "司徒正美55" )) //true |
alert(/^[\u4e00-\u9fa5]+$/.test( "正则表达式" )) //true |
alert(/^[\u4e00-\u9fa5]+$/.test( "正则表达式&*@@" )) //false |
alert(/\d{6}/.test( "123456" )) //true |
alert(/[ruby]{2}/.test( "rr" )) //true |
alert(/[ruby]{2}/.test( "ru" )) //true |
alert(/[ruby]{2}/.test( "ry" )) //true |
/[\u4e00-\u9fa5]/用于匹配单个汉字。
贪婪量词,惰性量词与支配性量词
贪婪量词,上面提到的全部简单量词。就像成语中说的巴蛇吞象那样,一口吞下整个字符串,发现吞不下(匹配不了),再从后面一点点吐出来(去掉最后一个字符,再看这时这个整个字符串是否匹配,不断这样重复直到长度为零)
隋性量词,在简单量词后加问号。因为太懒了,先吃了前面第一个字符,若是不饱再捏起多添加一个(发现不匹配,就读下第二个,与最初的组成一个有两个字符串的字符串再尝试匹配,若是再不匹配,再吃一个组成拥有三个字符的字符串……)。其工做方式与贪婪量词相反。
支配性量词,在简单量词后加加号。上面两种都有个不断尝试的过程,而支配性量词却只尝试一次,不合口味就算了。就像一个出身高贵居支配地位的公主。但你也能够说它是最懒量词。因为javascript不支持,因此它连出场的机会也没有了。
// var re3 = /.*+bbb/g;//支配性,javascript不支持,IE与全部最新的标准浏览器都报错 |
alert(re1.test( "abbbaabbbaaabbbb1234" )+ "" ); //true |
alert(re1.exec( "abbbaabbbaaabbbb1234" )+ "" ); //null |
alert( "abbbaabbbaaabbbb1234" .match(re1)+ "" ); //abbbaabbbaaabbbb |
alert(re2.test( "abbbaabbbaaabbbb1234" )+ "" ); //true |
alert(re2.exec( "abbbaabbbaaabbbb1234" )+ "" ); //aabbb |
alert( "abbbaabbbaaabbbb1234" .match(re2)+ "" ); //abbb,aabbb,aaabbb |
分组
到目前为止,咱们只能一个字符到匹配,虽然量词的出现,能帮助咱们处理一排密紧密相连的同类型字符。但这是不够的,下面该轮到小括号出场了,中括号表示范围内选择,大括号表示重复次数。小括号容许咱们重复多个字符。
alert(/(dog){2}/.test( "dogdog" )) //true |
alert( "baddad" .match(/([bd]ad?)*/)) //baddad,dad |
alert( "mon and dad" .match(/(mon( and dad)?)/)) //mon and dad,mon and dad, and dad |
反向引用
反向引用标识由正则表达式中的匹配组捕获的子字符串。每一个反向引用都由一个编号或名称来标识,并经过“\编号”表示法进行引用。
alert(RegExp.$1); //990000 |
alert(/(dog)\1/.test( "dogdog" )) //true |
var newNum = num.replace(/(\d{4}) (\d{4})/, "$2 $1" ); |
候选
继续在分组上作文章。在分组中插入管道符(“|”),把它划分为两个或多个候多项。
var reg = /(red|black|yellow)!!/; |
alert(reg.test( "red!!" )) //true |
alert(reg.test( "black!!" )) //true |
alert(reg.test( "yellow!!" )) //true |
非捕获性分组
并非全部分组都能建立反向引用,有一种特别的分组称之为非捕获性分组,它是不会建立反向引用。反之,就是捕获性分组。要建立一个非捕获性分组,只要在分组的左括号的后面紧跟一个问号与冒号就好了。
题目,移除全部标签,只留下innerText!
var text = html.replace(/<(?:.|\s)*?>/g, "" ); |
注意:javascript不存在命名分组
前瞻
继续在分组内作文章。前瞻与后瞻其实都属于零宽断言,但javascript不支持后瞻。
零宽断言 |
正则 |
名称 |
描述 |
(?=exp) |
正向前瞻 |
匹配exp前面的位置 |
(?!exp) |
负向前瞻 |
匹配后面不是exp的位置 |
(?<=exp) |
正向后瞻 |
匹配exp后面的位置不支持 |
(?<!exp) |
负向后瞻 |
匹配前面不是exp的位置不支持 |
正向前瞻用来检查接下来的出现的是否是某个特定的字符集。而负向前瞻则是检查接下来的不该该出现的特定字符串集。零宽断言是不会被捕获的。
var reBed = /(bed(?=room)) ///在咱们捕获bed这个字符串时,抢先去看接下来的字符串是否是room |
alert(reBed.test(str1)); //true |
alert(RegExp.$2 === "" ) //true |
alert(reBed.test(str2)) //false |
var reBed = /(bed(?!room))/ //要来它后面不能是room |
alert(reBed.test(str1)) //false |
alert(reBed.test(str2)) //true |
题目,移除hr之外的全部标签,只留下innerText!
var text = html.replace(/<(?!hr)(?:.|\s)*?>/ig, "" ) |
alert(text) //Ruby Louvre<hr/>by 司徒正美 |
边界
一个要与字符类合用的东西。
边界 |
正则 |
名称 |
描述 |
^ |
开头 |
注意不能紧跟于左中括号的后面 |
$ |
结尾 |
|
\b |
单词边界 |
指[a-zA-Z_0-9]以外的字符 |
\B |
非单词边界 |
|
题目,设计一个字符串原型方法,实现首字母大写!
String.prototype.capitalize = function () { |
return this .replace(/^\w/, function (s) { |
alert(a.capitalize()) //Ruby |
单词边界举例。要匹配的东西的前端或未端不能为英文字母阿拉伯字数字或下横线。
var str = "12w-eefd&efrew" ; |
alert(str.match(/\b\w+\b/g)) //12w,eefd,efrew |
实例属性 |
描述 |
global |
是当前表达式模式首次匹配内容的开始位置,从0开始计数。其初始值为-1,每次成功匹配时,index属性都会随之改变。 |
ignoreCase |
返回建立RegExp对象实例时指定的ignoreCase标志(i)的状态。若是建立RegExp对象实例时设置了i标志,该属性返回True,不然返回False,默认值为False。 |
lastIndex |
是当前表达式模式首次匹配内容中最后一个字符的下一个位置,从0开始计数,常被做为继续搜索时的起始位置,初始值为-1, 表示从起始位置开始搜索,每次成功匹配时,lastIndex属性值都会随之改变。(只有使用exec()或test()方法才会填入,不然为0) |
multiLine |
返回建立RegExp对象实例时指定的multiLine标志(m)的状态。若是建立RegExp对象实例时设置了m标志,该属性返回True,不然返回False,默认值为False。 |
source |
返回建立RegExp对象实例时指定的表达式文本字符串。 |