正则表达式系列基本用法

正则对象

生成正则对象

有两种方法可以创建并得到一个正则表达式对象

字面量声明方式

var reg = /abc/g;

显式创建

var reg =new RegExp("abc", "g");

可选项标记

上文生成正则对象中最后有用到g,其实这就是生成正则的可选项标记,可以组合使用,如下

g 全文匹配,即匹配一个成功后,如果没有结束,会继续匹配

i 忽略大小写匹配

m 多行匹配

可以通过如下示例进一步了解

/g的用法

加上/g后会全局搜索所有匹配结果

var str = 'abcabcabc';

// ['abc', index: 0, input: "abcabcabc"]

str.match(/abc/);

// ['abc', 'abc', 'abc']

str.match(/abc/g);

/i的用法

加上/i后大小写一视同仁,很多时候,如果不加/i,很容易出现大小写不匹配的错误(比如去除字符串中script标签时)

var str = 'Script';

// null

str.match(/script/);

// ["Script", index: 0, input: "Script"]

str.match(/script/i);

/m的用法

多行匹配比较少用到,但在一些场景下不得不用,比如下例中,只有\m多行匹配的情况下才能正常的将每一行的行尾(\n形成不同的行)的数字替换成#,否则默认会认为只有一行

var str = 'a1\nb1\nc1\nd1\ne1\nf1';

/**

* a1

* b1

* c1

* d1

* e1

* f#

*/

str.replace(/\d+$/g, '#')

/**

* a#

* b#

* c#

* d#

* e#

* f#

*/

str.replace(/\d+$/mg, '#')

子表达式

正则表达式中,用圆括号包围的就是子表达式(子模式)

一般,在match的非全局匹配中或exec中,在匹配完整个正则表达表达式后,都会再次匹配子表达式

另外,在字符串的replace与split中,子表达式也会经常用到

var str = 'a1.b2.c3.d4';

// 第一个结果是 整个正则表达式的匹配,之后则分别是子表达式的匹配

/(\w)(\d)[.]/.exec(str); // ["a1.", "a", "1", index: 0, input: "a1.b2.c3.d4"]

正则的test()

pattern.test(str);

检索字符串中是否存在指定模式,匹配成功则返回true,否则返回false

var str = 'abcdefg';

/^abc/.test(str); // true

/^abce/.test(str); // false

正则的compile()

reg.compile(pattern);

编译正则表达式,编译之后正则的执行速度会提高

编译的时候也可以改变检索模式或者改变可选项标识

var str = 'abcdabcdabc';

var reg = /abc/;

reg.test(str); // true

reg.compile();

reg.test(str); // true,仅仅是被编译,没有被改变

reg.compile(/aB/);

reg.test(str); // false,匹配式改成了aB,因此不匹配

reg.compile(/aB/i);

reg.test(str); // true,改成了aB并且忽略大小的形式,因此匹配

正则的exec()

pattern.exec(str);

在字符串中检索特定的模式,匹配成功则返回找到的值,否则返回null

有两种情况

第一种:非全局匹配

如果没有找到,则返回null

找到则返回一个数组,arr[0]是匹配结果,余下元素是arr[0]中匹配圆括号中子表达式的结果,以及最后的index和input

而且非全局模式中,不会保存index,也就是说不管匹配多少次,结果都是一样的

var str = 'a1.b2.c3.d4';

var reg1 = /(\w)(\d)[.]/;

reg1.exec(str); // ["a1.", "a", "1", index: 0, input: "a1.b2.c3.d4"]

reg1.exec(str); // ["a1.", "a", "1", index: 0, input: "a1.b2.c3.d4"]

reg1.exec(str); // ["a1.", "a", "1", index: 0, input: "a1.b2.c3.d4"]

/abc/.exec(str); // null

第二种:g全局匹配

正则的exec全局匹配可以保存index,并且下一次继续匹配时,将不会是重新从0开始,而是从保存的index开始

var str = 'a1.b2.c3.d4';

var reg2 = /(\w)(\d)[.]/g;

reg2.exec(str); // ["a1.", "a", "1", index: 0, input: "a1.b2.c3.d4"]

reg2.exec(str); // ["b2.", "b", "2", index: 3, input: "a1.b2.c3.d4"]

reg2.exec(str); // ["c3.", "c", "3", index: 6, input: "a1.b2.c3.d4"]

/abc/.exec(str); // null

字符串的正则应用

上文中提到的都是正则对象上的方法,但实际上,JS的String对象也支持正则表达式

字符串的match()

match是字符串中最常用的方法

str.match(pattern);

如果pattern中有g,代表全局匹配,则返回的数组包含所有匹配结果

如果无g,则返回的数组的第1个元素(arr[0])是第一个匹配结果,余下元素是arr[0]中匹配圆括号中子表达式的结果

var str = 'a.b2.c3.d445.e';

str.match(/\d[.]/); // ["2.", index: 3, input: "a.b2.c3.d445.e"]

// 非全局匹配下,并且有圆括号子表达式,先匹配整个正则表达式一次

// 然后在匹配结果中再匹配子表达式

str.match(/(\d)[.]/); // ["2.", "2", index: 3, input: "a.b2.c3.d445.e"]

// g 模式下是对整个正则表达式(包括圆括号子表达式)进行全局匹配

str.match(/(\d)[.]/g); // ["2.", "3.", "5."]

字符串的replace()

字符串中用来快速替换的方法,有多种用法

第一种情况

str.replace(str1, str2);

第一个参数是字符串,那么返回的结果只有str1被替换成str2了

var str = 'a.b2.c3.d4';

// 只会替换 . 一次

str.replace('.', '#'); // a#b2.c3.d445.e

第二种情况

str.replace(pattern, str2);

第一个参数是正则表达式,此时如果是g全局匹配模式,会替换所有的匹配结果为str2,否则只会替换一个匹配结果

var str = 'a.b2.c3.d4';

str.replace(/[.]/, '#'); // a#b2.c3.d445.e

str.replace(/[.]/g, '#'); // a#b2#c3#d445#e

另外此模式下,str2还可以使用一些有特殊含义的特殊字符,例如

var str = 'a1.b2.c3.d4';

// $2和$1分别代表第2个,第1个子表达式

str.replace(/(\w)(\d)[.]*/g, '$2$1~'); // 1a~2b~3c~4d~

str2中可用的特殊字符表

字符             替换

$1,$2,...,$99   匹配第1~99个pattern中的圆括号子表达式的文本

$&               匹配pattern的子串

$`                匹配子串的左边文本

$'                匹配子串的右边文本

$$               美元符号

第三种情况

str.replace(pattern, func);

这种模式下,第二个参数为一个函数,func将会在每一个匹配结果中调用,func返回的字符串将作为替换文本,func接收的参数,第一个是匹配pattern的字符串,之后的参数(可能是多个)是匹配该pattern中某个圆括号子表达式的字符串,在这之后的参数就是index(匹配结果的位置),再之后就是完整的input

var str = a1.b2.c3.d4;

// 最终结果为: 1a~2b~3c~4d~

str.replace(/(\w)(\d)[.]*/g, function(word, child1, child2, index, input) {

console.log(word); // 依次打印的是a1. b2. c3. d4

console.log(child1); // 依次打印的是a b c d

console.log(child2); // 依次打印的是1 2 3 4

console.log(index); // 依次打印的是0 3 6 9

console.log(input); // 每次都是打印 a1.b2.c3.d4

return child2 + child1 + '~';

});

字符串的search()

返回第1个与patten匹配的字符串子串的起始位置,如果找不到,则返回-1,不支持全局检索,也就是说,会省略g

var str = 'abcdefg1234567';

str.search(/efg/); // 4

str.search(/efg/g); // 4

str.search(/aabc/g); // -1

字符串的split()

split方法可以让一个字符串分割成数组,同样忽略g

str.split(pattern, limit); // pattern为字符串或正则

将str拆分为子串组成的数组,子串中不包括pattern(特例除外)。limit是可选参数,指定返回的数组的最大长度

特例: 如果pattern包含圆括号子表达式,则匹配这个圆括号子表达式的子串(不是匹配整个正则)会包含在返回数组中

var str = 'a1.b2.c3.d4';

str.split(/\d[.]/); // ["a", "b", "c", "d4"]

// 包含了圆括号的子串,返回数组中出现了匹配圆括号的子串

str.split(/(\d)[.]/); // ["a", "1", "b", "2", "c", "3", "d4"]

var str2 = '.b2.c3.';

// 3.后面没有字符了,但是由于它符合匹配,所以后续结果中会多一个""

str2.split(/\d[.]/); // [".b", "c", ""]

// 同上,只不过圆括号中的内容也在返回结果中

str.split(/(\d)[.]/); // [".b", "2", "c", "3", ""]

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,519评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,842评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,544评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,742评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,646评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,027评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,513评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,169评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,324评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,268评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,299评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,996评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,591评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,667评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,911评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,288评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,871评论 2 341

推荐阅读更多精彩内容

  • RegExp 三大方法本文的RegExp采用直接量语法表示:/pattern/attributes。attribu...
    恩德_b0c2阅读 449评论 0 0
  • 什么是正则表达式?如何创建正则表达式正则表达式常用的方法字符串中的正则表达式常用的正则表达式假设用户需要在HTML...
    greenlift阅读 801评论 0 0
  • re模块手册 本模块提供了和Perl里的正则表达式类似的功能,不关是正则表达式本身还是被搜索的字符串,都可以...
    喜欢吃栗子阅读 3,967评论 0 13
  • 这本书足够出名,也足够经典,我也足够吃了苦头,啃到深夜。亦苦亦甜。文章以丰富资料支撑,化以三言两语简短的案例而附注...
    你好乔纳森阅读 400评论 0 3
  • 不知道是怎么了,总会有些这样那样的困惑,人与人之间也会因为这样那样的事情略显尴尬。 清早换了一套轻熟的连衣裙,同事...
    沸水里的鱼阅读 416评论 0 1