字母文本指的是普通文本如"abcde"可匹配字符串中任何包含"abcde"的字符串。
元字符则更加灵活运用通用的表达式匹配所有符合此表达式规律的字符串。
[ ]——从中选择一个字符匹配
中间支持的类型:单词字符([ae])、非单词字符([!?,;@#$*])、字母范围([A-Z])、数字范围([0])
eg.
| 正则表达式 | 可匹配字符串 |
| [ae]ffect | affect,effect |
(此例中"[ae]"为元字符,"ffect"为字母文本)
注意:1.要在字符类中匹配连字符,那么把连字符号作为第一个字符列出即可。
2.可以在单个正则表达式中包含多个字符类。
eg. [01][0-9]:[0-5][0-9][ap]m 可以用来匹配如12:59pm格式的所有时间
^——排除某些字符(在[ ]中表此意,还可表示字符串的开头)
eg.
| 正则表达式 | 可匹配字符串 | 不可匹配字符串 |
| m[^a]t | met,mit,m&t…… | mat |
可以使用的特殊字符:
- \t ——匹配制表符
- \r ——匹配硬回车符
- \f ——匹配换页符
- \n——匹配换行符
描述表示字符类的元字符:
- . ——匹配任何除了\n以外的字符(或者在单行模式中的任何字符)
- \w ——匹配任何单词字符(任何字母或数字)
- \W ——匹配任何非单词字符(除了字母和数字以外的任何字符)
- \s ——匹配任何空白字符(包括空格、换行、制表符等)
- \S ——匹配任何非空白字符(除了空格、换行、制表符等的任何字符)
- \d ——匹配任何数字字符(0~9的数字)
- \D——匹配任何非数字字符(除了0~9以外的任何字符)
- ^ ——匹配字符串的开头(或者多行模式下行的开头)。
- $ ——匹配字符串的结尾,或者是字符串结尾“\n”之前的最后一个字符,或者是多行模式中的行结尾 。
- \A——匹配字符串的开头(忽略多行模式)
- \Z——匹配字符串的结尾或字符串结尾“\n”之前的最后一个字符(忽略多行模式)。
- \z——匹配字符串的结尾。
- \G——匹配当前搜索开始的位置。
- \b——匹配单词的边界。
- \B——匹配单词的非边界。
注意:
1.句点字符(.)特别有用。可以用它来表示任何一个字符。
eg.
| 正则表达式 | 可匹配字符串 |
| 01.17.84 | 01/17/84 , 01-17-84 , 01 17 84 , 01.17.84 |
2.可以使用\b匹配单词的边界
eg.
| 正则表达式 | 可匹配字符串 | 不可匹配字符串 |
| \blet\b | let | letter,hamlet |
3.\A和\z在确保字符串所包含的是某个表达式,而不是其他内容时很用。
eg.要判断Text控件是否包含单词"sophia",而不含任何额外的字符、换行符或者空白。
\Asophia\z
4.句点字符( . )具有特殊的含义,若要表示字母字符本身的含义,在前面加一个反斜杠:\.
三、匹配而选一的字符序列
|——匹配二选一
eg.
| 正则表达式 | 可匹配字符串 |
| col(o|ou)r | color,colour |
注意: \b(bill|ted)和\bbill|ted是不同的。
后者还可以匹配"malted"因为\b元字符只应用于"bill"。
- * ——匹配0次或多次
- + ——匹配1次或多次
- ? ——匹配0次或1次
- {n} ——恰好匹配n次
- {n,} ——至少匹配n次
- {n,m}——至少匹配n次,但不多于m次
eg.
| 正则表达式 | 可匹配字符串 |
| brothers? | brother,brothers |
eg.
| 正则表达式 | 可匹配字符串 |
| \bp\d{3,5}\b | 以p开头,且后跟3~5个数字结尾 |
注意:也可以把量词与()一起使用,以便把该量词应用到整个字母序列。
eg.
| 正则表达式 | 可匹配字符串 |
| (The)?school is beautiful. | school is beautiful,The school is beautiful. |
有些量词是贪婪的(greedy).他们会尽可能多的匹配字符。
如量词* 匹配0个或多个字符。假设要匹配字符串中任何HTML标签。你可能会用如下正则表达式:
<.*>
现有字符串A <i>quantifier</i> can be <big> greedy</big>
结果<.*>把 <i>quantifier</i> can be <big> greedy</big>都匹配上了。
要解决该问题,需要与量词一起使用一个特殊的非贪婪字符“?”,因此表达式变化如下:
<.*?>
这样就可以正确匹配<i>、</i>、<big>、</big>。
?能强制量词尽可能少地匹配字符,?还可以用在以下几个量词中:
- *? ——非贪婪的量词*
- +? ——非贪婪的量词+
- ?? ——非贪婪的量词?
- {n}? ——非贪婪的量词{n}
- {n,}? ——非贪婪的量词{n,}
- {n,m}?——非贪婪的量词{n,m}
捕获组(capture group)就像是正则表达式中的变量。捕获组可以捕获正则表达式中的字符模式,并且由正则表达式后面的编号或名称来引用改模式。
( ) ——用来捕获其中的字符串
\数字——用编号来引用
eg.
| 正则表达式 | 可匹配字符串 |
| (\w)(\w)\2\1 | abba |
注意:1.反向引用用来匹配html标签非常有效如<(\w+)></\1>可以匹配<table></table>等类似格式的标签。
2.默认情况下,只要使用圆括号,就会捕获圆括号内所包含的字符,可以使用n选项来禁用这个默认行为(在第7条里会详细介绍),或者添加?:到圆括号中。eg. (?:sophia) 或 (?n:sophia)此时不会捕获sophia。
(?<捕获组名称>)\k<捕获组名称>——用名称来引用
eg.
| 正则表达式 | 可匹配字符串 |
| (?<sophia>\w)abc\k<sophia> | xabcx |
注意:在替换模式中使用捕获组的格式略有不同,要用$1、$2等来按数值引用捕获,用${sophia}等名称来按名称引用捕获组
七、设置正则表达式的选项
eg.
- i ——所执行的匹配是不区分大小写的(.net中的属性为IgnoreCase)
- m——指定多行模式(.net中的属性为Multiline)
- n ——只捕获显示命名或编号的组(.net中的属性为ExplicitCapture)
- c ——编译正则表达式,这样会产生较快的执行速度,但启动会变慢(.net中的属性为Compiled)
- s ——指定单行模式(.net中的属性为SingleLine)
- x ——消除非转义空白字符和注释(.net中的属性为IgnorePatternWhitespace)
- r ——搜索从右到左进行(.net中的属性为RightToLeft)
- - ——表示禁用。
eg. (?im-r:sophia) 允许不区分大小写匹配sophia,使用多行模式,但禁用了从右到左的匹配。
注意:1.m会影响如何解析起始元字符(^)和结束元字符($)。在默认情况^和$只匹配整个字符串的开头,即使字符串包含多行文本。如果启用了m,那么它们就可以匹配每行文本的开头和结尾。
2.s会影响如何解析句点元字符(.)。通常一个句点能匹配除了换行符以外的所有字符。但在单行模式下,句点也能匹配一个换行符。
评论
-
> #最外层的右括号平衡组的一个最常见的应用就是匹配HTML,下面这个例子可以匹配嵌套的<div>标签:<div[^>]*>[^<>]*(((?'Open'<div[^>]*>)[^<>]*)+((?'-Open'</div>)[^<>]*)+)*(?(Open)(?!))</div>.
还有些什么东西没提到
我已经描述了构造正则表达式的大量元素,还有一些我没有提到的东西。下面是未提到的元素的列表,包含语法和简单的说明。你可以在网上找到更详细的参考资料来学习它们--当你需要用到它们的时候。如果你安装了MSDN Library,你也可以在里面找到关于.net下正则表达式详细的文档。
表7.尚未详细讨论的语法 代码/语法 说明
\a 报警字符(打印它的效果是电脑嘀一声)
\b 通常是单词分界位置,但如果在字符类里使用代表退格
\t 制表符,Tab
\r 回车
\v 竖向制表符
\f 换页符
\n 换行符
\e Escape
\0nn ASCII代码中八进制代码为nn的字符
\xnn ASCII代码中十六进制代码为nn的字符
\unnnn Unicode代码中十六进制代码为nnnn的字符
\cN ASCII控制字符。比如\cC代表Ctrl+C
\A 字符串开头(类似^,但不受处理多行选项的影响)
\Z 字符串结尾或行尾(不受处理多行选项的影响)
\z 字符串结尾(类似$,但不受处理多行选项的影响)
\G 当前搜索的开头
\p{name} Unicode中命名为name的字符类,例如\p{IsGreek}
(?>exp) 贪婪子表达式
(?<x>-<y>exp) 平衡组
(?im-nsx:exp) 在子表达式exp中改变处理选项
(?im-nsx) 为表达式后面的部分改变处理选项
(?(exp)yes|no) 把exp当作零宽正向先行断言,如果在这个位置能匹配,使用yes作为此组的表达式;否则使用no
(?(exp)yes) 同上,只是使用空表达式作为no
(?(name)yes|no) 如果命名为name的组捕获到了内容,使用yes作为表达式;否则使用no
(?(name)yes) 同上,只是使用空表达式作为no