教程
什么是正则表达式?
正则表达式是一组由字母和符号组成的特殊文本,它可以用来从文本中找出满足你想要的格式的句子。
元字符
元字符不代表他们本身的字面意思,他们都有特殊的含义。
元字符 | 描述 |
---|---|
. | 匹配任意单个字符除了换行符。 |
[ ] | 字符集。匹配方括号内的任意字符。 |
[^ ] | 否定的字符种类。匹配除了方括号里的任意字符 |
* | 匹配 >=0 个重复的在 * 号之前的字符。 |
+ | 匹配 >=1 个重复的 + 号前的字符。 |
? | 标记 ? 之前的字符为可选. |
{n,m} | 匹配 num 个大括号之间的字符 (n <= num <= m). |
(xyz) | 字符集,匹配与 xyz 完全相等的字符串. |
| | 或元字符,匹配符号前或后的字符. |
\ | 转义字符,用于匹配一些保留字符 `[ ] ( ) { } . * + ? ^ $ \ |
^ | 正则表达式的开始符号. |
$ | 正则表达式的结束符号. |
元字符 .
字符集 []
方括号用来指定一个字符集。
括号中的字符集不关心顺序。
表达式
[Tt]he
匹配the
和The
方括号的点号就表示点号
表达式
ar[.]
匹配ar.
字符串
否定字符集
一般来说 ^
表示一个字符串的开头,但当它用在方括号里面的开头时,它表示这个字符集是否定的。 例如,表达式 [^c]ar
匹配一个后面跟着 ar
的除了 c
的任意字符。
重复次数
后面跟着元字符 +
,*
or ?
的,用来指定匹配子模式的次数。 这些元字符在不同的情况下有着不同的含义。
元字符 *
*
号匹配在 *
之前的字符出现 >=0 次。 例如,表达式 a*
匹配 0个或者多个以 a 开头的字符。表达式 [a-z]*
匹配所有以小写字母开头的字符串。
*
元字符和 .
元字符搭配可以匹配所有的字符(.*
)。
*
和空格元字符 \s
连起来用,如表达式 \s*cat\s*
匹配以 0个或多个空格开头,中间是 cat,并且以 0个或多个空格结尾的字符串。
元字符 +
+
元字符匹配 +
号之前的字符出现 >=1 次。 例如表达式 c.+t
匹配以字母 c
开头以 t
结尾,中间跟着至少一个字符的字符串。
元字符 ?
元字符 ?
表示前面的字符为可选字符,即出现 0 次或 1 次。 例如,表达式 [T]?he
匹配字符串 he
和 The
。
元字符 {}
在正则表达式中 {}
是一个量词,常用来表示一个或一组字符可以重复出现的次数。 例如,表达式 [0-9]{2,3}
匹配最少 2 位最多 3 位 0~9 的数字。
可以省略第二个参数。 例如,[0-9]{2,}
匹配至少两位 0~9 的数字。
如果逗号也省略掉,则表示重复固定的次数。 例如,[0-9]{3}
匹配 3 位 0~9 的数字
元字符 ()
元字符 ()
表示将括号内的子模式绑定为一个组,将这个组作为一个新的子模式进行匹配。
例如之前说的 {}
是用来表示前面一个字符出现指定次数。但如果在 {}
前用括号将一串字符括起来,则表示整个括号内的字符重复 N 次。例如,表达式 (ab)*
匹配连续出现 0 个或多个 ab
。
还可以在 ()
中用 |
表示或。例如,(c|g|p)ar
匹配 car
或 gar
或 par
.
元字符 |
元字符 |
表示或,用作判断条件。
例如 (T|t)he|car
匹配 (T|t)he
或 car
。
元字符 \
元字符 \
用于转义紧跟其后的字符。用于转义 { } [ ] / \ + * . $ ^ | ?
这些特殊字符。如果想要匹配这些特殊字符则要在其前面加上反斜线 \
。
例如 .
是用来匹配除换行符外的所有单个字符的。如果想要匹配句子中的 .
则要写成 \.
。
例如 (f|c|m)at\.?
匹配以 f
或者 c
或者 m
开头,中间是 mt
,后面有 0 个或者 1 个 .
锚点
正则表达式中,想要匹配指定开头或结尾的字符串就要使用到锚点。^
指定开头,$
指定结尾。
元字符 ^
^
用来检查匹配的字符串是否在所匹配字符串的开头。
例如,在 abc
中使用表达式 ^a
会得到结果 a
。但如果使用 ^b
将匹配不到任何结果。因为在字符串 abc
中并没有以 b
开头的字符串。
例如,^(T|t)he
匹配以 The
或 the
开头的字符串。
元字符 $
与 ^
同理,元字符 $
用来检查匹配的字符串是否在所匹配字符串的结尾。
例如,(at\.)$
匹配以 at.
结尾的字符串。
简写字符集
正则表达式提供一些常用的字符集简写。如下:
简写 | 描述 |
---|---|
. | 除换行符外的所有单个字符 |
\w | 匹配所有字母数字,等同于 [a-zA-Z0-9_] |
\W | 匹配所有非字母数字,即符号,等同于: [^\w] |
\d | 匹配数字: [0-9] |
\D | 匹配非数字: [^\d] |
\s | 匹配所有空格字符,等同于: [\t\n\f\r\p{Z}] |
\S | 匹配所有非空格字符: [^\s] |
\f | 匹配一个换页符 |
\n | 匹配一个换行符 |
\r | 匹配一个回车符 |
\t | 匹配一个制表符 |
\v | 匹配一个垂直制表符 |
\p | 匹配 CR/LF(等同于 \r\n ),用来匹配 DOS 行终止符 |
零宽度断言
零宽度断言用来指定一个位置(零宽度),这个位置应该满足一定的条件(断言)。断言其实就相当于一个布尔值,只能是 0 或者 1。
由零宽度断言的定义可以看出来,它必须和和其他的表达式结合起来使用才具有意义,否则单纯的零宽度断言表达式的匹配结果就是个位置,什么字符都没有!
零宽度断言共四种:
符号 | 描述 |
---|---|
?= | 正先行断言 |
?! | 负先行断言 |
?<= | 正后发断言 |
?<! | 负后发断言 |
理解:
- 正/负:指是否满足条件(即断言),正(负)表示(不)满足。
- 先行/后发:指的是位置(最终的匹配结果)和条件(即断言)的前后关系,先行(后发)指位置在条件前(后)。
另外,零宽度断言都要用括号扩起来。
正先行断言 ?=
语法:表达式1(?=表达式2)
表示满足表达式1的字符串后面必须跟着满足表达式2的字符串,结果只返回满足匹配表达式1的字符串。
例如,表达式 (T|t)he(?=\sfat)
匹配 The
或者 the
,而且 The
和 the
后面紧跟着 空格fat
。
负先行断言 ?!
语法:表达式1(?=表达式2)
表示满足表达式1的字符串后面必须不能跟着满足表达式2的字符串,结果只返回满足匹配表达式1的字符串。
例如,表达式 (T|t)he(?!\sfat)
匹配 The
或者 the
,而且 The
和 the
后面不跟着 空格fat
。
正后发断言 ?<=
语法:(?<=表达式1)表达式2
表示满足表达式2的字符串前面面必须能跟着满足表达式1的字符串,结果只返回满足匹配表达式2的字符串。
例如,表达式 (?<=(T|t)he\s)(fat|mat)
匹配 fat
和 mat
,且其前面跟着 The空格
或 the空格
。
负后发断言 ?<!
语法:(?<!表达式1)表达式2
表示满足表达式2的字符串前面面必须不能跟着满足表达式1的字符串,结果只返回满足匹配表达式2的字符串。
例如,表达式 (?<!(T|t)he\s)(cat)
匹配 cat
,且其前不跟着 The
或 the
。
标志
标志也叫模式修正符,因为它可以用来修改表达式的搜索结果。 这些标志可以任意的组合使用,它也是整个正则表达式的一部分。
标志 | 描述 |
---|---|
i | 忽略大小写。 |
g | 全局搜索。 |
m | 多行修饰符:锚点元字符 ^ $ 工作范围在每行的起始。 |
忽略大小写(Case Insensitive)
修正符 i
用于忽略大小写。 例如,表达式 /The/gi
表示在全局搜索 The
,在后面的 i
将其条件修改为忽略大小写,则变成搜索 the
和 The
,g
表示全局搜索。
全局搜索(Global search)
修正符 g
常用于执行一个全局搜索匹配,即不仅仅返回第一个匹配的,而是返回全部。 例如,表达式 /.(at)/g
表示全局搜索 非换行符以外任意单个字符
后面跟着 at
。
多行修正符(Multiline)
多行修正符 m
用于执行一个多行匹配。
之前介绍的 ^
和 $
用于检查模式是否是在待检测字符串的开头或结尾。但我们如果想要它在每行的开头和结尾生效,我们需要用到多行修正符 m
。
例如,表达式 /at(.)?$/gm
表示小写字符 a
后跟小写字符 t
,末尾再跟上一个除换行符以外的任意字符,再根据 m
修正符,现在表达式匹配每行的结尾。
贪婪匹配与惰性匹配(Greedy vs lazy matching)
正则表达式默认采用贪婪匹配模式,在该模式下意味着会匹配尽可能长的子串。我们可以使用 ?
将贪婪匹配模式转化为惰性匹配模式。
例如 /(.*at)/
表示匹配以 at
结尾,前面有任意多个非换行字符的子字符串中长度最长的一个。
而 /(.*?at)/
表示匹配以 at
结尾,前面有任意多个非换行字符的子字符串中长度最短的一个。
实例
以某个字符串开始字符串,并且以另一个字符串结尾字符串
例如要搜索以 https://img.alicdn.com
开始,以 jpg_80x80.jpg
结束的子字符串。
1 | https:\/\/img\.alicdn\.com.*jpg_80x80\.jpg |