Github项目笔记|学习正则表达式的简单方法

0.jpg

学习正则表达式的简单方法

本篇为github上一个37.4K收藏项目的学习笔记。许多程序设计语言都支持利用正则表达式进行字符串操作。例如,在Perl建了一个功能强大的正则表达式引擎。正则表达式这个概念最初是由Unix中的工具软件普及开的。本篇则结合之前刚介绍过R中的stringr包,让大家更加直观的理解每个正则表达式的含义。

[TOC]

正则表达式是一种被用于从文本中检索符合某些特定模式的文本。正则表达式是从左到右来匹配一个字符串的。正则表达式可以被用来替换字符串中的文本、验证表单、基于模式匹配从一个字符串中提取字符串等等。

1. 基本匹配

正则表达式是用于在文本中检索字符串模式。例如正则表达式 cat,表示:字母 c 后面跟着一个字母 a,再后面跟着一个字母 t

library(tidyverse)
library(stringr)
x <- c("The fat cat sat on the mat.")
str_view_all(x,"cat")
1.0

正则表达式 123 会匹配字符串“123”。通过将正则表达式中的每个字符逐个与要匹配的字符串中的每个字符进行比较,来完成正则匹配。正则表达式通常区分大小写,因此正则表达式 Cat 与字符串“cat”不匹配。

x <- c("The fat cat sat on Cat.")
str_view_all(x,"Cat")
1.01

2. 元字符

元字符是正则表达式的基本组成元素。元字符在这里跟它通常表达的意思不一样,而是以某种特殊的含义去解释。有些元字符在写在方括号内时有特殊含义。
元字符如下:

元字符 描述
. 匹配除换行符以外的任意字符。
[ ] 字符类,匹配方括号中包含的任意字符。
[^ ] 否定字符类。匹配方括号中不包含的任意字符
* 匹配前面的子表达式零次或多次
+ 匹配前面的子表达式一次或多次
? 匹配前面的子表达式零次或一次,或指明一个非贪婪限定符。
{n,m} 花括号,匹配前面字符至少 n 次,但是不超过 m 次。
(xyz) 字符组,按照确切的顺序匹配字符 xyz。
| 分支结构,匹配符号之前的字符后面的字符。
\ 转义符,它可以还原元字符原来的含义,允许你匹配保留字符 <code>[ ] ( ) { } . * + ? ^ $ \ |</code>
^ 匹配行的开始
$ 匹配行的结束

2.1 英文句号

英文句号 . 是元字符的最简单的例子。元字符 . 可以匹配任意单个字符。它不会匹配换行符(\n)和新行的字符。例如正则表达式 .ar,表示:任意字符后面跟着一个字母 a,再后面跟着一个字母 r

x <- c("The car parked in the garage.")
str_view_all(x,".ar")
1.1

2.2 字符集

字符集也称为字符类。方括号被用于指定字符集。使用字符集内的连字符来指定字符范围。方括号内的字符范围的顺序并不重要。
例如正则表达式 [Tt]he,表示:大写 T 小写 t ,后跟字母 h,再后跟字母 e

x <- c("The car parked in the garage")
str_view_all(x,"[Tt]he")
1.2

然而,字符集中的英文句号表示它字面的含义。正则表达式 ar[.],表示小写字母 a,后面跟着一个字母 r,再后面跟着一个英文句号 . 字符。

x <- c("A garage is a good place to park a car.")
str_view_all(x,"ar[.]")
1.3

2.2.1 否定字符集

一般来说插入字符 ^ 表示一个字符串的开始,但是当它在方括号内出现时,它会取消字符集。例如正则表达式 [^c]ar,表示:除了字母 c 以外的任意字符,后面跟着字符 a,再后面跟着一个字母 r

x <- c("The car parked in the garage")
str_view_all(x,"[^c]ar")
1.4

2.3 重复

以下元字符 +*? 用于指定子模式可以出现多少次。这些元字符在不同情况下的作用不同。

2.3.1 星号

星号 * 表示匹配上一个匹配规则零次或多次。正则表达式 a* 表示小写字母 a 可以重复零次或者多次。但是它如果出现在字符集或者字符类之后,它表示整个字符集的重复
例如正则表达式 [a-z]*,表示:一行中可以包含任意数量的小写字母。

x <- c("The car parked in the garage #21.")
str_view_all(x,"[a-z]*")
1.5

星号 * 可以与元符号 . 用在一起,用来匹配任意字符串 .*。星号 * 可以与空格符 \s 一起使用,用来匹配一串空格字符。
例如正则表达式 \s*cat\s*,表示:零个或多个空格,后面跟小写字母 c,再后面跟小写字母 a,再在后面跟小写字母 t,后面再跟零个或多个空格。

x <- c("The fat cat sat on the cat.")
str_view_all(x,"\\s*cat\\s*") 
#值得注意的是R中的转义符为\\ 否则R会出现下列报错 如果要匹配就要写成"\\(\\)"
> str_view_all(x,"\s*cat\s*")
错误: 由""\s"开头的字符串中存在'\s',但没有这种逸出号
1.6

2.3.2 加号

加号 + 表示匹配上一个字符一次或多次。例如正则表达式 c.+t,表示:一个小写字母 c,后跟任意数量的字符,后跟小写字母 t

x <- c("The fat cat sat on the cat.")
str_view_all(x,"c.+t")
1.7

2.3.3 问号

在正则表达式中,元字符 ? 用来表示前一个字符是可选的。该符号匹配前一个字符零次或一次。
例如正则表达式 [T]?he,表示:可选的大写字母 T,后面跟小写字母 h,后跟小写字母 e

x <- c("The car parked in the garage")
str_view_all(x,"[T]he")
1.8
str_view_all(x,"[T]?he")
1.9

2.4 花括号

在正则表达式中花括号用于指定字符或一组字符可以重复的次数。例如正则表达式 [0-9]{2,3},表示:匹配至少 2 位数字但不超过 3 位(0 到 9 范围内的字符)。

x <- c("The number was 9.9907 but we rounded it off to 10.0 .")
str_view_all(x,"[0-9]{2,3}") #注意这里的0到9是[0-9],而不是[0,9]

我们可以省略第二个数字。例如正则表达式 [0-9]{2,},表示:匹配 2 个或更多个数字。如果我们也删除逗号,则正则表达式 [0-9]{2},表示:匹配正好为 2 位数的数字。

str_view_all(x,"[0-9]{2,}")
1.11
str_view_all(x,"[0-9]{2}")
1.12

2.5 字符组

字符组是一组写在圆括号内的子模式 (...)。正如我们在正则表达式中讨论的那样,如果我们把一个量词放在一个字符之后,它会重复前一个字符。但是,如果我们把量词放在一个字符组之后,它会重复整个字符组。
例如正则表达式 (ab)* 表示匹配零个或多个的字符串“ab”。我们还可以在字符组中使用元字符 |。例如正则表达式 (c|g|p)ar,表示:小写字母 cgp 后面跟字母 a,后跟字母 r

x <- c("The car parked in the garage")
str_view_all(x,"(c|g|p)ar")
1.13

2.6 分支结构

在正则表达式中垂直条 | 用来定义分支结构,分支结构就像多个表达式之间的条件。现在你可能认为这个字符集和分支结构的工作方式一样。但是字符集和分支结构巨大的区别是字符集只在字符级别上有作用,然而分支结构在表达式级别上依然可以使用。
例如正则表达式 (T|t)he|car,表示:匹配大写字母 T 小写字母 t,后面跟小写字母 h,后跟小写字母 e匹配小写字母 c,后跟小写字母 a,后跟小写字母 r

x <- c("The car parked in the garage")
str_view_all(x,"(T|t)he|car")
1.14

2.7 转义特殊字符

正则表达式中使用反斜杠 \ 来转义下一个字符。这将允许你使用保留字符来作为匹配字符 { } [ ] / \ + * . $ ^ | ?。在特殊字符前面加 \,就可以使用它来做匹配字符。
例如正则表达式 . 是用来匹配除了换行符以外的任意字符。现在要在输入字符串中匹配 . 字符,正则表达式 (f|c|m)at\.?,表示:小写字母 fc 或者 m 后跟小写字母 a,后跟小写字母 t,后跟可选的 . 字符。

x <- c("The fat cat sat on the cat.")
str_view_all(x,"(f|c|m)at\\.?")
1.15

2.8 定位符

在正则表达式中,为了检查匹配符号是否是起始符号或结尾符号,我们使用定位符。定位符有两种类型:

  • ^ 检查匹配字符是否是起始字符。
  • $检查匹配字符是否是输入字符串的最后一个字符。

2.8.1 插入符号

插入符号 ^ 符号用于检查匹配字符是否是输入字符串的第一个字符。如果我们使用正则表达式 ^a(如果 a 是起始符号)匹配字符串 abc,它会匹配到 a
但是如果我们使用正则表达式 ^b,它是匹配不到任何东西的,因为在字符串 abc 中“b”不是起始字符。
让我们来看看另一个正则表达式 ^(T|t)he,这表示:大写字母 T 或小写字母 t 是输入字符串的起始符号,后面跟着小写字母 h,后跟小写字母 e

x <- c("The car parked in the garage")
str_view_all(x,"(T|t)he")
1.16
str_view_all(x,"^(T|t)he")
1.17

2.8.2 美元符号

美元 $ 符号用于检查匹配字符是否是输入字符串的最后一个字符。例如正则表达式 (at\.)$,表示:小写字母 a,后跟小写字母 t,后跟一个 . 字符,且这个匹配器必须是字符串的结尾

x <- c("The fat cat. sat. on the cat.")
str_view_all(x,"(at\\.)")
1.18
str_view_all(x,"(at\\.)$")
1.19

3. 简写字符集

正则表达式为常用的字符集和常用的正则表达式提供了简写。简写字符集如下:

简写 描述
. 匹配除换行符以外的任意字符
\w 匹配所有字母和数字的字符:[a-zA-Z0-9_]
\W 匹配非字母和数字的字符:[^\w]
\d 匹配数字:[0-9]
\D 匹配非数字:[^\d]
\s 匹配空格符:[\t\n\f\r\p{Z}]
\S 匹配非空格符:[^\s]

4. 断言

后行断言(lookbehind)和先行断言(lookahead)有时候被称为断言,它们是特殊类型的 非捕获组(用于匹配模式,但不包括在匹配列表中)。当我们在一种特定模式之前或者之后有这种模式时,会优先使用断言。
例如我们想获取输入字符串 $4.44 and $10.88 中带有前缀 $ 的所有数字。我们可以使用这个正则表达式 (?<=\$)[0-9\.]*,表示:获取包含 . 字符且前缀为 $ 的所有数字。
以下是正则表达式中使用的断言:

符号 描述
?= 正向先行断言
?! 负向先行断言
?<= 正向后行断言
?<! 负向后行断言

4.1 正向先行断言

正向先行断言认为第一部分的表达式的后面必须是先行断言表达式。返回的匹配结果仅包含与第一部分表达式匹配的文本。要在一个括号内定义一个正向先行断言,在括号中问号和等号是这样使用的 (?=...)。先行断言表达式写在括号中的等号后面。
例如正则表达式 (T|t)he(?=\sfat),表示:匹配大写字母 T 或小写字母 t,后面跟字母 h,后跟字母 e
在括号中,我们定义了正向先行断言,它会引导正则表达式引擎匹配后面跟着 fatThethe

x <- c("The fat cat sat on the mat.")
str_view_all(x,"(T|t)he(?=\\sfat)")
1.20
str_view_all(x,"(T|t)he(?=\\smat)")
1.21

4.2 负向先行断言

当我们需要指定第一部分表达式的后面不跟随某一内容时,使用负向先行断言。负向先行断言的定义跟我们定义的正向先行断言一样,唯一的区别在于我们使用否定符号 ! 而不是等号 =,例如 (?!...)
我们来看看下面的正则表达式 (T|t)he(?!\sfat),表示:从输入字符串中获取全部 The 或者 the 且不匹配 fat 前面加上一个空格字符。

x <- c("The fat cat sat on the mat.")
str_view_all(x,"(T|t)he(?!\\sfat)")
1.22

4.3 正向后行断言

正向后行断言用于获取跟随在特定模式之后的所有匹配内容。正向后行断言表示为 (?<=...)

例如正则表达式 (?<=(T|t)he\s)(fat|mat),表示:从输入字符串中获取在单词 Thethe 之后的所有 fatmat 单词。

x <- c("The fat cat sat on the mat.")
str_view_all(x,"(?<=(T|t)he\\s)(fat|mat)")
1.23

4.4 负向后行断言

负向后行断言是用于获取不跟随在特定模式之后的所有匹配的内容。负向后行断言表示为 (?<!...)。例如正则表达式 (?<!(T|t)he\s)(cat),表示:在输入字符中获取所有不在 Thethe 之后的所有单词 cat

x <- c("The cat sat on cat.")
str_view_all(x,"(?<!(T|t)he\\s)(cat)")
1.24

5. 标记

标记也称为修饰符,因为它会修改正则表达式的输出。这些标志可以以任意顺序或组合使用,并且是正则表达式的一部分。

标记 描述
i 不区分大小写:将匹配设置为不区分大小写。
g 全局搜索:搜索整个输入字符串中的所有匹配。
m 多行匹配:会匹配输入字符串每一行。

5.1 不区分大小写

i 修饰符用于执行不区分大小写匹配。例如正则表达式 /The/gi,表示:大写字母 T,后跟小写字母 h,后跟字母 e。但是在正则匹配结束时 i 标记会告诉正则表达式引擎忽略这种情况。正如你所看到的,我们还使用了 g 标记,因为我们要在整个输入字符串中搜索匹配。

这个在R中我暂时不会实现。

<pre>
"The" => <a href="#learn-regex"><strong>The</strong></a> fat cat sat on the mat.
</pre>

<pre>
"/The/gi" => <a href="#learn-regex"><strong>The</strong></a> fat cat sat on <a href="#learn-regex"><strong>the</strong></a> mat.
</pre>

5.2 全局搜索

g 修饰符用于执行全局匹配(会查找所有匹配,不会在查找到第一个匹配时就停止)。
例如正则表达式 /.(at)/g,表示:除换行符之外的任意字符,后跟小写字母 a,后跟小写字母 t
因为我们在正则表达式的末尾使用了 g 标记,它会从整个输入字符串中找到每个匹配项。

<pre>
".(at)" => The <a href="#learn-regex"><strong>fat</strong></a> cat sat on the mat.
</pre>

<pre>
"/.(at)/g" => The <a href="#learn-regex"><strong>fat</strong></a> <a href="#learn-regex"><strong>cat</strong></a> <a href="#learn-regex"><strong>sat</strong></a> on the <a href="#learn-regex"><strong>mat</strong></a>.
</pre>

5.3 多行匹配

m 修饰符被用来执行多行的匹配。正如我们前面讨论过的 (^, $),使用定位符来检查匹配字符是输入字符串开始或者结束。但是我们希望每一行都使用定位符,所以我们就使用 m 修饰符。
例如正则表达式 /at(.)?$/gm,表示:小写字母 a,后跟小写字母 t,匹配除了换行符以外任意字符零次或一次。而且因为 m 标记,现在正则表达式引擎匹配字符串中每一行的末尾。

<pre>
"/.at(.)?$/" => The fat
cat sat
on the <a href="#learn-regex"><strong>mat.</strong></a>
</pre>

<pre>
"/.at(.)?$/gm" => The <a href="#learn-regex"><strong>fat</strong></a>
cat <a href="#learn-regex"><strong>sat</strong></a>
on the <a href="#learn-regex"><strong>mat.</strong></a>
</pre>

6. 常用正则表达式

  • 正整数^\d+$

  • 负整数^-\d+$

  • 电话号码^+?[\d\s]{3,}$

  • 电话代码^+?[\d\s]+(?[\d\s]{10,}$

  • 整数^-?\d+$

  • 用户名^[\w\d_.]{4,16}$

  • 字母数字字符^[a-zA-Z0-9]*$

  • 带空格的字母数字字符^[a-zA-Z0-9 ]*$

  • 密码^(?=^.{6,}$)((?=.*[A-Za-z0-9])(?=.*[A-Z])(?=.*[a-z]))^.*$

  • 电子邮件^([a-zA-Z0-9._%-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,4})*$

  • IPv4 地址^((?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?))*$

  • 小写字母^([a-z])*$

  • 大写字母^([A-Z])*$

  • 网址^(((http|https|ftp):\/\/)?([[a-zA-Z0-9]\-\.])+(\.)([[a-zA-Z0-9]]){2,4}([[a-zA-Z0-9]\/+=%&_\.~?\-]*))*$

  • VISA 信用卡号码^(4[0-9]{12}(?:[0-9]{3})?)*$

  • 日期(MM/DD/YYYY)^(0?[1-9]|1[012])[- /.](0?[1-9]|[12][0-9]|3[01])[- /.](19|20)?[0-9]{2}$

  • 日期(YYYY/MM/DD)^(19|20)?[0-9]{2}[- /.](0?[1-9]|1[012])[- /.](0?[1-9]|[12][0-9]|3[01])$

  • 万事达信用卡号码^(5[1-5][0-9]{14})*$


正则表达式虽然语法不多,但是却常常化繁为简。(说实在的,除了简单的,我敲完基本也就忘了,常用常新,如果大家只是偶尔用到,我觉得没必要如数家珍,用到了再去查查或者把这篇文章翻出来看看问题也不大。) 另外也有一些在线测试正则表达式的网页,大家可以拿来练手,如: https://tool.oschina.net/regex/#。如果觉得这篇文章对你有帮助的话,希望大家点个支持一下~

Reference

[1] https://github.com/cdoco/learn-regex-zh

[2] https://github.com/ziishaned/learn-regex

License

MIT © [Zeeshan Ahmed


往期内容

《R数据科学》学习笔记|Note9:使用stringr处理字符串(上)

《R数据科学》学习笔记|Note10:使用stringr处理字符串(下)

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

推荐阅读更多精彩内容