相关文章推荐
彷徨的机器人  ·  stringvar转变成str ...·  4 天前    · 
爱笑的高山  ·  Windows 10 - ...·  7 月前    · 
小胡子的眼镜  ·  Postgres On ...·  1 年前    · 
打篮球的海豚  ·  c++ - no crash if I ...·  1 年前    · 
卖萌的滑板  ·  有趣的 CSS - ...·  1 年前    · 
骑白马的开水瓶  ·  绿色云 | Oracle 中国·  1 年前    · 
python——正则表达式

python——正则表达式

官方中文文档: re --- 正则表达式操作 - Python 3.9.6 文档

  • 正则语法表

以下为常用语法,部分不常用语法(先行断言(lookahead)和后行断言(lookbehind) )见补充: 正则表达式的先行断言(lookahead)和后行断言(lookbehind) | 菜鸟教程

范例所用文本:

"<link rel="dns-prefetch" href="//wl.jd.com" />
<title>eastman - 商品搜索 - 京东</title>
<meta name="Keywords" content="eastman,京东eastman" />
<meta name="description" content="在京东找到了eastman1313件eastman的类似商品,其中包含了eastman价格、eastman评论、eastman导购、eastman图片等相关信息" />"
模式 描述 范例
^span 从整个字符串开头的地方匹配,字符串必须以span开头。在 MULTILINE 模式中也匹配换行后的首个符号。[^span] 指除span外的所有字符串 表达式:^<.*结果:['<link rel="dns-prefetch" href="//wl.jd.com" />']
span$ 匹配至整个字符串结尾处结束,字符串必须以span结尾,可与^结合使用.在 MULTILINE 模式下也会匹配换行符之前的文本。 表达式:\S\S\S\S\S\W/>$结果:['相关信息" />']
. 匹配所有字符,换行符除外 表达式:<ti........结果:['<title>east']
* 匹配零次或多次*前面的分组 表达式:\W\W搜索*结果:['商品搜索']
匹配零次或一次?前面的分组 表达式:me(\w)?结果:['met','me','met','me']
+ 匹配一次或多次+前面的分组,包括换行符。 表达式:[0-9]+结果:['1313']
| 管道,匹配多个表达式中任意一个 表达式:Keywords|京东easT结果:['Keywords']
() 分组,将括号内的内容作为一个整体,创建子表达式,子表达式可供获取 表达式:Keywords|京东(easT)?结果:['京东','Keywords','京东','京东']
[] 表示一个字符集,它常被用来指定一个字符类别,字符可以单个列出,也可以用“-”号分隔的两个给定字符来表示一个字符区间 表达式:[0-9]结果:['1','3','1','3']
{n} 匹配n次前边的表达式 表达式:[0-9]{3}结果:['131']
{n,} 匹配n次及n次以上前面的分组 表达式:[0-9]{1,}结果:['1313']
{,m} 匹配0次到m次前面的分组 表达式:[0-9]{,2}结果:
{n,m} 匹配至少n次,至多m次前边的表达式。m和n均为非负整数 表达式:[0-9]{1,3}结果:['131']
\w 匹配任意字母、数字或下划线,等价于 [A-Za-z0-9_] 表达式:</\w\w\w结果:['<\tit']
\W 匹配除字母、数字、下划线以外的任意字符 表达式:\W\W\W\w\w\w\w\w\w\w\d结果:['找到了eastman1']
\d 匹配0-9的数字,等价于[0-9] 表达式:\d...结果:['1313']
\D 匹配除数字以外的所有字符 表达式:\D.\d结果:['an1']
\s 匹配空格、换行符、制表符及回车、换页符,即空白字符,等价于[\t\n\f\r\v] 表达式:>\s<tit结果:['>\n<tit']
\S 匹配任意非空白字符,等价于[^\t\n\f\r\v] 表达式:>\S\S\S\S\S\S\S\s\S结果:['>eastman -']
\b 匹配一个单词边界,也就是指单词和空格间的位置 ‘er\b’ 可以匹配"never" 中的 ‘er’,但不能匹配 “verb” 中的 ‘er’。
\B 匹配非单词边界。‘ er\B’ 能匹配 “verb” 中的 ‘er’,但不能匹配 “never” 中的 ‘er’。
(?#...)或# 注释 print(re.findall('\d+ (?#匹配数字)',text,re.X))#['1313']print(re.findall('\d+ #匹配数字',text,re.X))#['1313']
(?imx) 正则表达式包含三种可选标志:i, m, 或 x 。只影响括号中的区域。应置于表达式最前方。 print(re.findall('(?x)\d+ (?#匹配数字)',text))#['1313']
(?-imx) 正则表达式关闭 i, m, 或 x 可选标志。只影响括号中的区域。 print(re.findall('(?-x)\d+ (?#匹配数字)',text,re.X))#re.error: missing : at position 4
(?: re) 类似 (...), 但是不表示一个组 print(re.findall('(?:\d+)',text))#['1313']
(?imx: re) 在括号中使用i, m, 或 x 可选标志 print(re.findall('(?x:\d+)',text))#['1313']
(?-imx: re) 在括号中不使用i, m, 或 x 可选标志 print(re.findall('(?-x:\d+)',text))#['1313']


  • 贪婪模式与非贪婪模式

python默认贪婪模式。

贪婪模式:在匹配结果有二义时,尽量匹配最大长度字符串。

'*', '+',和 '?' 修饰符都是 贪婪的,它们在字符串进行尽可能多的匹配。有时候并不需要这种行为。如果正则式 <.*> 希望找到 '<a> b <c>' ,它将会匹配整个字符串,而不仅是 '<a>'

在修饰符之后添加 ? 将使样式以非贪婪`方式进行匹配 ,尽量少的字符将会被匹配。 使用正则式 <.*?> 将会仅仅匹配 '<a>'


  • python中常用正则函数


re.match() 与 re.search()

语法:re.match(pattern, string, flags=0)

尝试从字符串的起始位置匹配一个模式, 如果不是起始位置匹配成功的话,match()就返回none。

即便是 MULTILINE 多行模式, re.match() 也只匹配字符串的开始位置,而不匹配每行开始。

参数 描述
pattern 匹配的正则表达式
string 要匹配的字符串。
flags 标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等
匹配方法 描述
group(num=0) 匹配的整个表达式的字符串,group() 可以一次输入多个组号,在这种情况下它将返回一个包含那些组所对应值的元组
groups() 返回一个包含所有分组的字符串的元组,从 1 到 所含的分组号
start([group]) 方法用于获取分组匹配的子串在整个字符串中的起始位置(子串第一个字符的索引),参数默认值为 0
end([group]) 方法用于获取分组匹配的子串在整个字符串中的结束位置(子串最后一个字符的索引+1),参数默认值为 0
span([group]) 方法返回 (start(group), end(group)

实例:

import re
text = '''<link rel="dns-prefetch" href="//wl.jd.com" />
<title>eastman - 商品搜索 - 京东</title>
mo = re.match('<\w\w\w',text)
print(mo)#<re.Match object; span=(0, 4), match='<lin'>
print(mo.group())#<lin
print(mo.start())#0
print(mo.span())#(0, 4)
mo1 = re.match('rel',text)#'rel'不在该字符串起始位置,所以返回None
print(mo1)#None
'''一般常这样用 :'''
content = 'Hello 123 456 welcome to tuling'
result = re.match('^Hello(.*)ng$', content)
print(result)#<re.Match object; span=(0, 31), match='Hello 123 456 welcome to tuling'>
print(result.group())#Hello 123 456 welcome to tuling
print(result.groups())#(' 123 456 welcome to tuli',)
print(result.group(1))#123 456 welcome to tuli
print(result.span(1))#(5, 29)

语法:re.search(pattern, string, flags=0)

扫描整个字符串并 返回第一个成功的匹配 。如果搜索完了还没有找到,就返回 None。

匹配方法与re.match()相同

import re
text = '''<link rel="dns-prefetch" href="//wl.jd.com" />
<title>eastman - 商品搜索 - 京东</title>
<meta name="Keywords" content="eastman,京东eastman" />
<meta name="description" content="在京东找到了eastman1313件eastman的类似商品,其中包含了eastman价格、eastman评论、eastman导购、eastman图片等相关信息" />
mo = re.search('^<meta',text,re.MULTILINE)# MULTILINE 多行模式中函数 match() 只匹配字符串的开始,但使用 search() 和以 '^' 开始的正则表达式会匹配每行的开始。
print(mo)#<re.Match object; span=(82, 87), match='<meta'>
print(mo.group())#<meta
print(mo.span())#(82, 87)
mo1 = re.search('^meta',text,re.MULTILINE)#使用'^'限制匹配到字符串的首位
print(mo1)#Non

re.match只匹配字符串的开始 ,如果字符串开始不符合正则表达式,则匹配失败,函数返回None;而 re.search匹配整个字符串 ,直到找到一个匹配。

re.findall() 与 re.finditer()

语法:findall(string[, pos[, endpos]])

在字符串中找到正则表达式所匹配的所有子串,并返回一个列表,如果没有找到匹配的,则返回空列表。

注意: match 和 search 是匹配一次 findall 匹配所有

参数 描述
string 待匹配的字符串
pos 可选参数,指定字符串的起始位置,默认为 0
endpos 可选参数,指定字符串的结束位置,默认为字符串的长度。
import re
text = '''<link rel="dns-prefetch" href="//wl.jd.com" />
<title>eastman - 商品搜索 - 京东</title>
<meta name="Keywords" content="eastman,京东eastman" />
<meta name="description" content="在京东找到了eastman1313件eastman的类似商品,其中包含了eastman价格、eastman评论、eastman导购、eastman图片等相关信息" />
mo = re.findall('\w\weastman',text)
print(mo)#['京东eastman', '到了eastman', '3件eastman', '含了eastman'

语法:re.finditer(pattern, string, flags=0)

re.finditer 和 re.findall 类似,在字符串中找到正则表达式所匹配的所有子串,并 把它们作为一个迭代器返回 。迭代器中的每一个元素都是一个Match对象

import re
text = '''<link rel="dns-prefetch" href="//wl.jd.com" />
<title>eastman - 商品搜索 - 京东</title>
<meta name="Keywords" content="eastman,京东eastman" />
<meta name="description" content="在京东找到了eastman1313件eastman的类似商品,其中包含了eastman价格、eastman评论、eastman导购、eastman图片等相关信息" />
mo = re.finditer('\w\weastman',text)
print(mo) #<callable_iterator object at 0x000001B47F1CD448>
for i in mo:
    print(i)
    print(i.group())
#输出结果:
#<re.Match object; span=(121, 130), match='京东eastman'>
#京东eastman
#<re.Match object; span=(173, 182), match='到了eastman'>
#到了eastman
#<re.Match object; span=(185, 194), match='3件eastman'>
#3件eastman
#<re.Match object; span=(203, 212), match='含了eastman'>
#含了eastma

split()

语法:re.split(pattern, string, maxsplit=0, flags=0)

按照能够匹配的子串将字符串分割后返回列表。

如果 maxsplit 非零, 最多进行 maxsplit 次分隔, 剩下的字符全部返回到列表的最后一个元素。

参数 描述
pattern 匹配的正则表达式
string 要匹配的字符串。
maxsplit 分隔次数,maxsplit=1 分隔一次,默认为 0,不限制次数。
flags 标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等。
import re
a = re.split(r'\W+', 'Words, words, words')
print(a)#['Words', 'words', 'words']

如果在表达式中有圆括号,那么 除被 圆括号中的表达式匹配掉的字符 外的字符 也会作为元素被包含在列表里(含空白字符)

b = re.split(r'(\W+)', 'Words,\n,words, words')
print(b)#['Words', ',\n,', 'words', ', ', 'words']

如果分隔符里有捕获组合,并且匹配到字符串的开始或结尾,那么结果将会以一个空字符串开始或结尾

c = re.split(r'\W+', '.words, words。')
print(c)#['', 'words', 'words', '']

对于一个找不到匹配的字符串而言,split 不会对其作出分割

d = re.split('\s', 'helloworld')
print(d)#['helloworld']

样式的空匹配仅在与前一个空匹配不相邻时才会拆分字符串

e = re.split('1*','helloworld')
print(e)#['', 'h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', '']

re.sub()

语法:re.sub(pattern, repl, string, count=0, flags=0)

返回通过使用 repl 替换在 string 最左边非重叠出现的 pattern 而获得的字符串。 如果样式没有找到,则不加改变地返回 string。

repl 可以是字符串或函数。 如repl为字符串,则其中任何反斜杠转义序列都会被处理 。 也就是说,\n 会被转换为一个换行符,\r 会被转换为一个回车附,依此类推。

参数 描述
pattern 匹配的正则表达式
repl 替换的字符串,也可为一个函数。
string 要被查找替换的原始字符串
count 模式匹配后替换的最大次数,默认 0 表示替换所有的匹配。
import re
phone = "2004-959-559 # 这是一个国外电话号码"
# 删除字符串中的 Python注释
num = re.sub(r'#.*$', "", phone)
print("电话号码是:%s"%num) #电话号码是:2004-959-559 
# 删除非数字(-)的字符串
num = re.sub(r'\D', "", phone)
print("电话号码是:%s"%num) #电话号码是:20049595

compile()

语法:re.compile(pattern[, flags])

用于编译正则表达式,生成一个正则表达式( Pattern )对象,供 match() 和 search() 这两个函数使用。

match() 或 search() 函数匹配成功后,返回的是一个match对象。该对象可使用如group()、start()、span()等方法。

import re