導讀:正則在各語言中的使用是有差別的,本文以 Python 3 爲基礎。本文主要講述的是正則的語法,對於 re 模塊不作過多描述,只會對一些特殊地方作提示。python
不少人以爲正則很難,在我看來,這些人必定是沒有用心。其實正則很簡單,根據二八原則,咱們只須要懂 20% 的內容就能夠解決 80% 的問題了。我曾經有幾年幾乎天天都跟正則打交道,剛接手項目的時候我對正則也是一無所知,花半小時百度了一下,而後寫了幾個 demo,就開始正式接手了。三年多時間,我用到的正則鮮有超出我最初半小時百度到的知識的。正則表達式
(1)經常使用元字符bash
語法 | 描述 |
---|---|
\b | 匹配單詞的開始或結束 |
\d | 匹配數字 |
\s | 匹配任意不可見字符(空格、換行符、製表符等),等價於[ \f\n\r\t\v]。 |
\w | 匹配任意 Unicode 字符集,包括字母、數字、下劃線、漢字等 |
. | 匹配除換行符(\n)之外的任意字符 |
^ 或 \A | 匹配字符串或行的起始位置 |
$ 或 \Z | 匹配字符串或行的結束位置 |
(2)限定詞(又叫量詞)app
語法 | 描述 |
---|---|
* | 重複零次或更屢次 |
+ | 重複一次或更屢次 |
? | 重複零次或一次 |
{n} | 重複 n 次 |
{n,} | 重複 n 次或更屢次 |
{n,m} | 重複 n 到 m 次 |
(3)經常使用反義詞工具
語法 | 描述 |
---|---|
\B | 匹配非單詞的開始或結束 |
\D | 匹配非數字 |
\S | 匹配任意可見字符, [^ \f\n\r\t\v] |
\W | 匹配任意非 Unicode 字符集 |
[^abc] | 除 a、b、c 之外的任意字符 |
(4)字符族ui
語法 | 描述 |
---|---|
[abc] | a、b 或 c |
[^abc] | 除 a、b、c 之外的任意字符 |
[a-zA-Z] | a 到 z 或 A 到 Z |
[a-d[m-p]] | a 到 d 或 m 到 p,即 [a-dm-p](並集) |
[a-z&&[def]] | d、e 或 f(交集) |
[a-z&&[^bc]] | a 到 z,除了 b 和 c:[ad-z](減去) |
[a-z&&[^m-p]] | a 到 z,減去 m 到 p:[a-lq-z](減去) |
以上即是正則的基礎內容,下面來寫兩個例子看下:this
s = '123abc你好'
re.search('\d+', s).group()
re.search('\w+', s).group()
複製代碼
結果:spa
123
123abc你好
複製代碼
是否是很簡單? code
修飾符在各語言中也是有差別的。cdn
Python 中的修飾符:
修飾符 | 描述 |
---|---|
re.A | 匹配 ASCII字符類,影響 \w, \W, \b, \B, \d, \D |
re.I | 忽略大小寫 |
re.L | 作本地化識別匹配(這個極少極少使用) |
re.M | 多行匹配,影響 ^ 和 $ |
re.S | 使 . 匹配包括換行符(\n)在內的全部字符 |
re.U | 匹配 Unicode 字符集。與 re.A 相對,這是默認設置 |
re.X | 忽略空格和 # 後面的註釋以得到看起來更易懂的正則。 |
(1)re.A
修飾符 A
使 \w
只匹配 ASCII 字符,\W
匹配非 ASCII 字符。
s = '123abc你好'
re.search('\w+', s, re.A).group()
re.search('\W+', s, re.A).group()
複製代碼
結果:
123abc
你好
複製代碼
可是描述中還有 \d
和 \D
,數字不都是 ASCII 字符嗎?這是什麼意思?別忘了,還有 全角和半角!
s = '0123456789' # 全角數字
re.search('\d+', s, re.U).group()
複製代碼
結果:
0123456789
複製代碼
(2)re.M 多行匹配的模式其實也不經常使用,不多有一行行規整的數據。
s = 'aaa\r\nbbb\r\nccc'
re.findall('^[\s\w]*?$', s)
re.findall('^[\s\w]*?$', s, re.M)
複製代碼
結果:
['aaa\r\nbbb\r\nccc'] # 單行模式
['aaa\r', 'bbb\r', 'ccc'] # 多行模式
複製代碼
(3)re.S 這個簡單,直接看個例子。
s = 'aaa\r\nbbb\r\nccc'
re.findall('^.*', s)
re.findall('^.*', s, re.S)
複製代碼
結果:
['aaa\r']
['aaa\r\nbbb\r\nccc']
複製代碼
(4)re.X 用法以下:
rc = re.compile(r""" \d+ # 匹配數字 # 和字母 [a-zA-Z]+ """, re.X)
rc.search('123abc').group()
複製代碼
結果:
123abc
複製代碼
注意,用了
X
修飾符後,正則中的全部空格會被忽略,包括正則裏面的本來有用的空格。若是正則中有須要使用空格,只能用\s
代替。
(5)(?aiLmsux) 修飾符不只能夠代碼中指定,也能夠在正則中指定。(?aiLmsux)
表示了以上全部的修飾符,具體用的時候須要哪一個就在 ? 後面加上對應的字母,示例以下,(?a)
和 re.A
效果是同樣的:
s = '123abc你好'
re.search('(?a)\w+', s).group()
re.search('\w+', s, re.A).group()
複製代碼
結果是同樣的:
123abc
123abc
複製代碼
當正則表達式中包含能接受重複的限定符時,一般的行爲是(在使整個表達式能獲得匹配的前提下)匹配儘量多的字符。
s = 'aabab'
re.search('a.*b', s).group() # 這就是貪婪
re.search('a.*?b', s).group() # 這就是懶惰
複製代碼
結果:
aabab
aab
複製代碼
簡單來講:
*
、+
、{n,}
這些表達式屬於貪婪;*?
、+?
、{n,}?
這些表達式就是懶惰(在貪婪的基礎上加上 ?
)。語法 | 描述 |
---|---|
(exp) | 匹配exp,並捕獲文本到自動命名的組裏 |
(?Pexp) | 匹配exp,並捕獲文本到名稱爲 name 的組裏 |
(?:exp) | 匹配exp,不捕獲匹配的文本,也不給此分組分配組號 |
(?P=name) | 匹配以前由名爲 name 的組匹配的文本 |
注意:在其餘語言或者網上的一些正則工具中,分組命名的語法是
(?<name>exp)
或(?'name'exp)
,但在 Python 裏,這樣寫會報錯:This named group syntax is not supported in this regex dialect。Python 中正確的寫法是:(?P<name>exp)
示例一:
分組可讓咱們用一條正則提取出多個信息,例如:
s = '姓名:張三;性別:男;電話:138123456789'
m = re.search('姓名[::](\w+).*?電話[::](\d{11})', s)
if m:
name = m.group(1)
phone = m.group(2)
print(f'name:{name}, phone:{phone}')
複製代碼
結果:
name:張三, phone:13812345678
複製代碼
示例二:
(?P<name>exp)
有時仍是會用到的, (?P=name)
則不多狀況下會用到。我想了一個 (?P=name)
的使用示例,給你們看下效果:
s = ''' <name>張三</name> <age>30</age> <phone>138123456789</phone> '''
pattern = r'<(?P<name>.*?)>(.*?)</(?P=name)>'
It = re.findall(pattern, s)
複製代碼
結果:
[('name', '張三'), ('age', '30'), ('phone', '138123456789')]
複製代碼
語法 | 描述 |
---|---|
(?=exp) | 匹配exp前面的位置 |
(?<=exp) | 匹配exp後面的位置 |
(?!exp) | 匹配後面跟的不是exp的位置 |
(?<!exp) | 匹配前面不是exp的位置 |
注意:正則中經常使用的前項界定
(?<=exp)
和前項否認界定(?<!exp)
在 Python 中可能會報錯:look-behind requires fixed-width pattern,緣由是 python 中 前項界定的表達式必須是定長的,看以下示例:
(?<=aaa) # 正確
(?<=aaa|bbb) # 正確
(?<=aaa|bb) # 錯誤
(?<=\d+) # 錯誤
(?<=\d{3}) # 正確
複製代碼
這大概是最複雜的正則表達式了。語法以下:
語法 | 描述 |
---|---|
(?(id/name)yes|no) | 若是指定分組存在,則匹配 yes 模式,不然匹配 no 模式 |
此語法極少用到,印象中只用過一次。
如下示例的要求是:若是以 _ 開頭,則以字母結尾,不然以數字結尾。
s1 = '_abcd'
s2 = 'abc1'
pattern = '(_)?[a-zA-Z]+(?(1)[a-zA-Z]|\d)'
re.search(pattern, s1).group()
re.search(pattern, s2).group()
複製代碼
結果:
_abcd
abc1
複製代碼
Python 中的 re.findall
是個比較特別的方法(之因此說它特別,是跟我經常使用的 C# 作比較,在沒看註釋以前我想固然的掉坑裏去了)。咱們看這個方法的官方註釋:
Return a list of all non-overlapping matches in the string.
If one or more capturing groups are present in the pattern, return
a list of groups; this will be a list of tuples if the pattern
has more than one group.
Empty matches are included in the result.
複製代碼
簡單來講,就是
看下面的例子:
s = 'aaa123bbb456ccc'
re.findall('[a-z]+\d+', s) # 不包含分組
re.findall('[a-z]+(\d+)', s) # 包含一個分組
re.findall('([a-z]+(\d+))', s) # 包含多個分組
re.findall('(?:[a-z]+(\d+))', s) # ?: 不捕獲分組匹配結果
複製代碼
結果:
['aaa123', 'bbb456']
['123', '456']
[('aaa123', '123'), ('bbb456', '456')]
['123', '456']
複製代碼
零寬斷言中講到 Python 中前項界定必須是定長的,這很不方便,可是配合 findall 有分組時只取分組結果的特性,就能夠模擬出非定長前項界定的效果了。
其實正則就像是一個數學公式,會背公式不必定會作題。但其實這公式一點也不難,至少比學校裏學的數學簡單多了,多練習幾回也就會了。