|
|
正则表达式之基本概念# S) D5 u5 G# L- B5 ?! z) s- T
) @" \+ B2 r: k* V在我们写页面时,往往需要对表单的数据比如账号、身份证号等进行验证,而最有效的、用的最多的便是使用正则表达式来验证。那什么是正则表达式呢?
( q- M$ |5 ?/ z3 g正则表达式(Regular Expression)是用于描述一组字符串特征的模式,用来匹配特定的字符串。 它的应用非常广泛,特别是在字符串处理方面。其常见的应用如下:
( P) k- B% a- z( l 7 }7 T; a) z+ D; n
- 验证字符串,即验证给定的字符串或子字符串是否符合指定的特征,例如,验证是否是合法的邮件地址、验证是否是合法的HTTP地址等等。
- Y0 }' y1 r5 A# M' p5 } - 查找字符串,从给定的文本当中查找符合指定特征的字符串,这样比查找固定字符串更加灵活。
6 u8 }8 b# e) E - 替换字符串,即查找到符合某特征的字符串之后将之替换。 + y" ` E0 d9 o+ O0 f4 i# `
- 提取字符串,即从给定的字符串中提取符合指定特征的子字符串。" ]! {( \( O" @% _4 j3 l8 ]
第一部分:正则表达式之工具
+ \4 X# V2 F! x4 Z" e2 ?; i N! M; q% M$ m5 V2 X' |& {
正所谓工欲善其事必先利其器! 所以我们需要知道下面几个主要的工具:
- \, v }, [$ f4 i7 @: I yhttp://www.regexpal.com/ 这个网站中,我们可以在线测试正则表达式。3 H* x" g# m F+ m
http://regexr.com/ 这个网站更为推荐,它自身还包括了一个实例使我们直接测试。
( W f! ]- C+ p, G& z: |第二部分:正则表达式之元字符
# W- E8 N) j* X6 S- k
4 q6 S$ D$ E4 ]& X2 r: b3 x正则表达式中元字符恐怕是我们听得最多的了。元字符(Metacharacter)是一类非常特殊的字符,它能够匹配一个位置或者字符集合中的一个字符。 如.、\w等都是元字符。, U5 t" F/ }( Q: x
刚刚说到,元字符既可以匹配位置,也可以匹配字符,那么我们就可以通过此来将元字符分为匹配位置的元字符和匹配字符的元字符。
0 S) h5 Y& |+ Z4 i' N5 pA匹配位置的元字符---^、$、\b
/ h1 w- `2 L- M5 _" Y" V" C' d) d. h
9 E& q) x; Y+ @. N8 e5 C即匹配位置的元字符只有^(脱字符号)、$(美元符号)和\b这三个字符。分别匹配行的开始、行的结尾以及单词的开始或结尾。它们匹配的都只是位置。& B* a+ w2 N9 ^/ u1 m
1.^匹配行的开始位置
4 W' M) o% o( h3 J L5 v
& |3 E4 C, E. S: ]% g6 Y如^zzw匹配的是以"zzw"为行开头的"zzw"(注意:我这里想要表达的是:尽管加了一个^,它匹配的仍是字符串,而不是一整行!),如果zzw不是作为行开头的字符串,则它不会被匹配。0 C& {( S0 q: M6 y; {
1 K$ Z0 Y5 [% f5 K
2.$匹配行的结尾位置
+ `3 a& K+ O- E4 E. e l( S t$ f+ E6 k. h
如zzw$匹配的是以"zzw"为行结尾的"zzw"(同样,这里$只是匹配的一个位置,那个位置是零宽度,而不是一整行),如果zzw不是作为行的结尾,那么它不会被匹配。# [8 F1 e2 o y4 l6 `3 G
/ G! u0 r c0 j6 f
于是结合^和$我们就不难猜测^zzw$匹配的是某行中只有zzw这个字符串了。
& X. [; Z# ~% U* X
# n& a, ~' `) [1 x: c! e. r; u8 l而^$匹配的则是一个空行,这个空行中不包含任何字符串。
& P) E) E) n+ x( w1 v2 ?. F3. \b匹配单词的开始或结束) I; {/ D; O! i
0 k/ K* @" V8 W8 Q9 Z6 O* [6 M
如 \bzzw匹配的是zzw之前是空格符号、标点符号或换行符号的zzw(注意:但\b匹配的仅是一个零宽度的位置,而不会匹配空格、标点符号或换行符号)。1 O$ L, F- \ v& Q- Y2 V
而zzw\b匹配的是zzw之后是空格符号、标点符号或换行符号的zzw(同样,\b匹配的是一个零宽度的位置)。
% u/ Z3 K) ~5 `% R显然 \bzzw\b匹配的就是zzw的前后必须是空格符号、标点符号或换行符号的zzw。
. G1 s/ u9 h1 h! `
; e. w0 v$ U" t3 s# m; bB 匹配字符的元字符---. 、\w、\W、\s、\S、\d、\D3 }3 w3 ?' H1 x- [9 w: y ~
! [ O3 b* w$ [7 k1 h
即匹配字符的元字符共有七个。: t# U; m, ~, A3 v: E
其中.(点号)表示匹配除换行符之外的任意字符;
( P7 V) V( a5 Z, E, h4 z 6 f8 y( X5 y, K. w4 ?
- \w 匹配单词字符(不仅仅是字母,还有下划线、数字和汉字);\W匹配任意的非单词字符(注意,与\w刚好相反);
. w4 ?& ^/ \! o7 z - \s匹配的是任意的空白字符(如空格、制表符、换行符、中文全角空格等); 7 r: f* A ?$ u2 w; F. C* x
- \S匹配的是任意的非空白字符(注意:刚好和\s相反);
. N1 ^0 @. o1 ^) x; `# z - \d匹配任意的数字;
) A3 O' B: b2 P- g: g C) o- A# V8 K - \D匹配任意的非数字字符(注意:刚好和\d相反)。
2 s7 d: F: y+ [5 @' j) m1 @ 例子如下所示:
% s+ v, G2 g5 h
3 G: j" I( G$ j+ t& w( J: K7 z上面的四个例子从左到右依次表示:& e) O' [8 `$ v1 N7 J& G
9 |8 @3 s" x& ^, ~ K' @( O: U* ]$ a
- ^.$表示匹配一行中的唯一一个任意的非换行符的字符
0 C; z$ W! Z( r2 I! }7 Q - \ba\w\w\w\w\w\w\w\w\w\b表示匹配以字母a开始的后面有9个字母字符的单词。 (注意:其中的a并不是元字符,就是一个普通的字符,我们称之为字符串字面值(string literal)-所谓字符串字面值,就是字面上看起来是什么就是什么)
9 p" a v! s' q3 ~/ g( W2 K& V* W4 H V - \b\w\w\w\d\d\d\d\D\b表示匹配以3个字母字符开始后面紧跟着四个数字字符且最后一个不是数字字符的单词。$ w: V; o* U. L: c. z0 o% `1 D0 j
第三部分:正则表达式之文字匹配
) u3 P5 T" v4 ^: @- m8 R i9 R& M& O- `; |9 F! ], l% d j
这一部分我们将通过字符类、字符转义、反义这几个部分来学习。, }( @2 {8 d' D( l y
A、字符类
4 A ?& ~& i6 A i
8 T9 G. C1 A+ M" H字符类是正则表达式中的“迷你”语言,可以在[]中定义。
( Z9 @0 k1 b' C/ w3 g最简单的字符类可以由[]和几个简单的字母组成。比如[aeiou]可以匹配aeiou五个字母中的任意一个。[0123456]可以匹配0123456这七个数字中的任意一个。而可以匹配到HTML标记中的中的任意一个标记。而[bhc]at就会匹配字符串bat、hat、cat。也就是说字符类[]中的多个字符只会匹配其中一个。
]7 M* U! b) c- g6 U3 @- j但是显然[0123456]、这样的表示太麻烦,需要写的很多,所以我们可以使用 -连字符)来简写之,如[0-6]和。于是可知[0-9]和\d的作用是一样的。[a-z]可以表示所有的小写字母,[A-Z]可以表示所有的大写字母。[a-zA-Z]可以表示所有的大写字母和小写字母。
, b. H) g/ O$ b8 q+ w: K% o值得注意的是 -(连字符) 只有在字符类中的中间位置时才是“到”的意思。而[-b]5中-没有在两个字符之间,所以它表示的是-5或者b5。) d$ u* @9 J8 |0 w9 Q
除此之外,我们知道^时只匹配行的开头,但是如果^出现在字符类中的第一个位置,那么它表示否定该字符类。如[^123]表示匹配不是数字1或2或3的其他任意字符。[^-]表示匹配不是-的任意字符。 ?% f6 @: p3 J U, k' M) r: O
由此我们也可以发现:在字符类中使用元字符(-、^等)时,不需要进行转义运算。7 v0 N, N S! J; Q% ~
更常用的有[^aeiou]匹配元音之外的字符、[0-9a-zA-Z_]匹配任何数字、字母(大写和小写)和下划线,这等同于\w、[^0-9a-zA-Z_]匹配任何非数字、字母(大写和小写)和下划线,这等同于\W。9 q, x4 l, a+ e' ^) b. D
$ O7 {7 m- k( Z8 U' M, Q
B、字符转义
4 X" T( q5 C& c" Z5 A) R! V
( ?; S. a( t2 w, m* |我们知道元字符如&、^、.等都表示着特殊的涵义,如果我们希望把他们看作一般的字符去匹配字符串,并且恰好他们又都不在字符类(如[&]中),我们就需要使用\(反斜杠)进行转义了。
( y+ ^9 t/ i" h" a如我们可以使用www.jb51.net来匹配www.jb51.net。 我们可以用\*来匹配字符串中的*(通配符)。 我们还可以通过\\来匹配\。举例如下所示:
; C" q* S! x& H/ ?4 f' p
/ A5 @4 C& E; \2 T# h' jC.反义
" h3 ^6 Z. Y! n/ H/ c
1 ^7 _* V+ ]4 d# ]. h实际上我们在前面已经说过了,这里再说一遍是因为希望可以引起注意,即^再字符类中的最前面时表示对这个字符类中的字符表示否定。如a[^b]匹配a和a后面不是b的字符。又如]>表示匹配的字符。举例如下:8 t8 b+ E8 o8 Z) U J1 H$ ~
: p1 R4 L# e9 O* w& ^+ S; S3 r3 Z第四部分:正则表达式之限定符
8 X+ F* f# |3 t+ E! S" ?& e% _$ M' r9 U8 Y% N
什么是限定符呢? 我们知道,在前面的例子中,我曾经使用过\ba\w\w\w\w\w\w\w\w\w\b表示匹配以字母a开始的后面有9个字母字符的单词,显然这样写是十分麻烦的,如果我们能把\w这些重复的用简单的形式来写就好了~ 没错,限定符就是干这个的,利用限定符我们可以重新写成\ba\w{9}\b。对,就是这么简单! 下面让我们了解更多吧。2 u# ]) o4 _1 V9 B
, w) i6 C: m i1 b" J
- {n} 表示重复n次,如\w{5}表示匹配个单词字符。
# ?! g: C/ o( ^) q - {n,} 表示重复至少n次,如\w{5}表示匹配至少5个单词字符,也可以是6个,7个……
9 S, j4 e. o$ R2 X0 K+ m - {n,m} 表示重复至少n次,最多m次,如\w{5,10} 表示匹配至少5个,最多10个单词字符。
$ ~! u! ?8 K, i" D' |- p; h) j - * 表示重复至少0次。 等同于{0,} ,即hu*t可以匹配ht或hut或huut或huuut……
+ f( ~( y7 b B; `7 z; ~ - + 表示重复至少1次。 等同于{1,},即hu+t可以匹配 hut或huut或huuut……
$ h; y$ _3 K4 ?6 a* K2 I! D - ? 表示重复0次或1次。等同于{0,1},即colou?r表示匹配color或者是colour。8 Z$ F" I6 `* `# K. C O* u. a2 Z
显然,上面的限定符所指的i安定都是限定的前面的某一个字符。1 U6 z; N+ N% t! `3 ^( Q' x
但是,如果我们在上面的限定符之后加上一个?呢 ,这时我们称之为懒惰限定符。 相应地,我们称上面几种匹配为贪婪匹配。1 f2 X' u! D, o2 X. L! Q9 B
$ d# `. n. p& q3 P
- {n}? 等同于{n} 9 f9 P$ |5 m/ }4 [0 E
- {n,}? 尽可能少的使用重复,但至少使用n次 ( C: z0 h6 O9 X: o! v. ~& E, F
- {n,m}? 重复n次到m次之间,但要尽可能少的使用重复。 + q6 `# M0 f8 A# X7 R" I
- *? 尽可能少的使用重复的第一个匹配 ( c& y1 I+ P' `/ ]! L) l
- +? 尽可能少的使用重复,但至少使用1次 - q( y( L3 I' T! K4 {) n2 O j% ?
- ?? 使用零次重复(如果有可能)或者一次重复% d! g! N/ w/ l$ @
比如:对于aabab这个要 匹配的字符串而言,使用a.*b会匹配aabab,而如果使用a.*?b就会匹配aab和ab,而不是匹配所有。
0 y$ ^1 [3 c3 n. S5 n: d$ U/ Z# q4 |
第五部分:正则表达式之字符的运算" Q# s. q% a1 t* O- N
, D1 @. S$ v! y7 \
字符的运算包括替换、分组和反向引用,下面我将逐项介绍。3 A# l/ Y: Z* r% h- _
A 替换
- q6 j- ~$ g8 D$ O
/ R2 l' E! j; @8 W6 J. z' I什么是替换呢? 显然就是指一个不行,我用另一个来替换,比如0\d{3}-\d{7}|0\d{2}-\d{8} 就表示匹配前4位为区号、后7位为本地号码的电话号码,亦可匹配前3位为区号、后8位为本地号码的电话号码。| 表示的就是替换了。 又如[Jj]ack和Jack|jack的匹配效果是一样的都是用来匹配Jack或jack的。也就是说,替换|是或运算的关系。3 G# t+ j0 }6 T1 D0 ~; ^9 k9 p: N
在一般的或运算中0 0结果为0, 0 1结果为1, 1 0结果为1, 1 1结果为1。那么在正则表达式中同样是这样,如果一个都匹配不上就不匹配;如果有一个能匹配上就匹配一个;如果两个都能匹配上,就匹配两个。 举例如下所示:
" s. B* X E( A: y
# e; Y: \8 i Q3 mB.分组. }# l/ b$ r6 ~% a
6 }) j. z8 M% Q6 c1 t4 D
正则表达式中,分组也是一个非常重要的概念。看似复杂,实际上分组就是使用"("和")",即左圆括号和右圆括号将某些字符括起来看成一个整体来处理。
# `% o' E5 p x, C% V9 C比如我们希望匹配abcabc。如果是abc{3}匹配的就是abccc,这不能达到预期,所以我们可以对abc分组,即(abc){3}就可以匹配到我们想要的字符串了。$ t A4 P* X1 h1 g4 v2 C+ E1 k3 e
又如(\d{1,3}\.){3}\d{1,3}也用了分组的运算方式,它可以用来匹配简单的IP地址,如下所示:+ w& |3 I7 {, R
+ Q- k" M7 \1 F4 [& Y9 VC 反向引用2 ^( b* |. ~6 x2 j' U/ @& ]
% _: f2 J8 N; Z0 k. v' b6 t上面我们通过()可以进行分组,而分组的同时,每一个组被自动赋予了一个组号,该组号可以代表该组的表达式。" T+ E' H4 s9 |1 H3 L
编组的规则是:从左到右、以分组的左括号"("为标志,第一个分组的组号为1,第二个分组的组号为2,以此类推。! {; s$ U* Y; D- ^5 n8 E! M
这时,反向引用就派上用场啦。我们就可以用它来反向引用使用()括起来的字符组了。具体怎么引用呢?规则如下:; f0 V" B/ ^# h- ?4 u8 h
& z7 R8 E# q. M5 q3 y% H- \数字,使用数字命名的反向引用。注:这个是通用的一种方式。 3 O! R) D3 P& H. ^7 u! b
- \k,使用指定命名的反向引用。注:这个是.NET Frameword支持的一种方式。9 d# s# l9 n+ e2 a" {
下面的几个例子是使用数字命名的反向引用:
8 f' X7 C- Z5 b( l3 R9 y! `8 ]# @# ~; Z4 J
其中我们可以看到第一个和第二个匹配的并不相同---第一个匹配的是两个任意单词字符组成的单词,而第二个由于是使用了反向引用,那么它就必须是两个重复单词字符组成的单词。. Z) e5 o6 P: f3 D9 k
最后一个我使用了两个分组,根据分组规则可知\w{3}\d{2}是第一个分组,\d{2}是第二个分组。 同样注意:反向引用的是同样的字符,如www55www5566,最后两位不同,这就无法正确匹配了。
3 }9 ?( s5 t, Q3 W使用指定命名(也就是自定义命名)的反向引用% d) P5 e( @1 D" Z+ y
对于上面的第二个例子,我们用自定义命名的反向引用可以写成\b(?\w)\k\b或者是\b(? 'myName' \w)\k\b。 本想举例试一下,结果都提示错误,可能是上面提到的两个在线网站不支持吧~5 _1 O z" f3 M' D& H! u
当然,如果我们希望只是将之看作一个整体,而不希望给其编号,可以使用(?:expression)的方式。如下所示:$ c! J% L" a# j% a+ h+ w3 I
- l o1 ^5 U+ R" n, Q) w. d @
另外,下面几个也是常用的分组:
( B- r: z% t, C& n
2 o( j* ^# Y2 w& r- (?=expression)匹配字符串expression前面的位置 ' t" _% }+ `9 Z8 h
- (?!expression)匹配后面不是字符串expression的位置 ) u. Y( {8 z1 ?$ U) v5 R7 C% Z' f
- (?(?>expression)只匹配字符串expression一次
7 m* Q7 w( u. R, c' A* A D.零宽度断言
) |. r5 W' B( H4 ~
! Y: |) x& _& q& X之前介绍的^和$都是匹配的一个满足一定条件的位置。这里把满足的一个条件成为断言或零宽度断言。
* I ^/ P& z) }3 K: D常用的有:
+ e0 L6 Z5 ?, S1 H
7 E* |4 L6 X/ x4 t+ X3 t. g- ^ 匹配行的开始位置 & }( K; ]0 v! E2 V& }3 }# p9 Y
- $ 匹配行的结束位置
! i+ c7 X% c3 F$ s$ f9 O9 \/ t! a% a3 o - \A 匹配必须出现在字符串的开头 C3 _( L* ?+ u
- \Z 匹配必须出现在字符串的结尾或字符串结尾处的\换行符好n之前
5 O/ ^8 v3 Z# Z- J/ n( e - \z 匹配必须出现在字符串的结尾
# w. m) n6 m/ h$ d. M - \G 匹配必须出现在上一个匹配结束的地方 6 M- d* w I& m1 z* o# }! d! Z, N
- \b 匹配字符的开始或结束位置
- W O$ M6 t! D+ D2 O+ M1 j" B - \B 匹配不是在字符的开始或结束位置# T( a5 t' h: s' o
之前所提到的(?=expression)、(?!expression)、(?其中(?=expression)称为零宽度预测先行断言,它断言自身位置的后面能够匹配表达式expression。如\b\w+(?=ed\b)可以匹配以字符串ed结尾的单词的前面部分,如reseted中的reset。
0 C0 A% y" I* D0 s8 ?7 U/ {% X5 s6 @; V) F3 w( ]) Z$ |: M
其中(? |
本帖子中包含更多资源
您需要 登录 才可以下载或查看,没有账号?立即注册
×
|