正则表达式是一个很强大的字符串处理工具,几乎任何关于字符串的操作都可以使用正则表达式来完成,作为一个爬虫工作者,每天和字符串打交道,正则表达式更是不可或缺的技能,正则表达式的在不同的语言中使用方式可能不一样,不过只要学会了任意一门语言的正则表达式用法,其他语言中大部分也只是换了个函数的名称而已,本质都是一样的。下面,我来介绍一下python中的正则表达式是怎么使用的。
首先,python中的正则表达式大致分为以下几部分:
元字符
模式
函数
re 内置对象用法
分组用法
环视用法
所有关于正则表达式的操作都使用 python 标准库中的 re 模块。
一、元字符 (参见 python 模块 re 文档)
. 匹配任意字符(不包括换行符)
^ 匹配开始位置,多行模式下匹配每一行的开始
$ 匹配结束位置,多行模式下匹配每一行的结束
* 匹配前一个元字符0到多次
+ 匹配前一个元字符1到多次
? 匹配前一个元字符0到1次
{m,n} 匹配前一个元字符m到n次
\ 转义字符,跟在其后的字符将失去作为特殊元字符的含义,例如\.只能匹配.,不能再匹配任意字符
[] 字符集,一个字符的集合,可匹配其中任意一个字符
| 逻辑表达式 或 ,比如 a|b 代表可匹配 a 或者 b
(…) 分组,默认为捕获,即被分组的内容可以被单独取出,默认每个分组有个索引,从 1 开始,按照”(“的顺序决定索引值
(?iLmsux) 分组中可以设置模式,iLmsux之中的每个字符代表一个模式,用法参见 模式 I
(?:…) 分组的不捕获模式,计算索引时会跳过这个分组
(?P…) 分组的命名模式,取此分组中的内容时可以使用索引也可以使用name
(?P=name) 分组的引用模式,可在同一个正则表达式用引用前面命名过的正则
(?#…) 注释,不影响正则表达式其它部分,用法参见 模式 I
(?=…) 顺序肯定环视,表示所在位置右侧能够匹配括号内正则
(?!…) 顺序否定环视,表示所在位置右侧不能匹配括号内正则
(?<=…) 逆序肯定环视,表示所在位置左侧能够匹配括号内正则
(?
regex = re.compile(“(?#注释)(?i)hello world!”)
print regex.match(s).group()
复制代码
L LOCALE, 字符集本地化。这个功能是为了支持多语言版本的字符集使用环境的,比如在转义符\w,在英文环境下,它代表[a-zA-Z0-9_],即所以英文字符和数字。如果在一个法语环境下使用,缺省设置下,不能匹配”é” 或 “ç”。加上这L选项和就可以匹配了。不过这个对于中文环境似乎没有什么用,它仍然不能匹配中文字符。
M MULTILINE,多行模式, 改变 ^ 和 $ 的行为
复制代码
s = ”’first line
second line
third line”’
regex_start = re.compile(“^\w+”)
print regex_start.findall(s)
regex_start_m = re.compile(“^\w+”, re.M)
print regex_start_m.findall(s)
regex_end = re.compile(“\w+$”)
print regex_end.findall(s)
regex_end_m = re.compile(“\w+$”, re.M)
print regex_end_m.findall(s)
复制代码
S DOTALL,此模式下 ‘.’ 的匹配不受限制,可匹配任何字符,包括换行符
复制代码
s = ”’first line
second line
third line”’
#
regex = re.compile(“.+”)
print regex.findall(s)
regex_dotall = re.compile(“.+”, re.S)
print regex_dotall.findall(s)
复制代码
X VERBOSE,冗余模式, 此模式忽略正则表达式中的空白和#号的注释,例如写一个匹配邮箱的正则表达式
email_regex = re.compile(“[\w+.]+@[a-zA-Z\d]+.(com|cn)”)
email_regex = re.compile(“”“[\w+.]+ # 匹配@符前的部分
@ # @符
[a-zA-Z\d]+ # 邮箱类别
.(com|cn) # 邮箱后缀 “”“, re.X)
U UNICODE,使用 \w, \W, \b, \B 这些元字符时将按照 UNICODE 定义的属性.
正则表达式的模式是可以同时使用多个的,在 python 里面使用按位或运算符 | 同时添加多个模式
如 re.compile(”, re.I|re.M|re.S)
每个模式在 re 模块中其实就是不同的数字
复制代码
print re.I
print re.L
print re.M
print re.S
print re.X
print re.U
复制代码
三、函数 (参见 python 模块 re 文档)
python 的 re 模块提供了很多方便的函数使你可以使用正则表达式来操作字符串,每种函数都有它自己的特性和使用场景,熟悉之后对你的工作会有很大帮助
compile(pattern, flags=0)
给定一个正则表达式 pattern,指定使用的模式 flags 默认为0 即不使用任何模式,然后会返回一个 SRE_Pattern (参见 第四小节 re 内置对象用法) 对象
regex = re.compile(“.+”)
print regex
这个对象可以调用其他函数来完成匹配,一般来说推荐使用 compile 函数预编译出一个正则模式之后再去使用,这样在后面的代码中可以很方便的复用它,当然大部分函数也可以不用 compile 直接使用,具体见 findall 函数
复制代码
s = ”’first line
second line
third line”’
#
regex = re.compile(“.+”)
print regex.findall(s)
print regex.search(s).group()
复制代码
escape(pattern)
转义 如果你需要操作的文本中含有正则的元字符,你在写正则的时候需要将元字符加上反斜扛 \ 去匹配自身, 而当这样的字符很多时,写出来的正则表达式就看起来很乱而且写起来也挺麻烦的,这个时候你可以使用这个函数,用法如下
复制代码
s = “.+\d123”
#
regex_str = re.escape(“.+\d123”)
print regex_str
for g in re.findall(regex_str, s):
print g
复制代码
findall(pattern, string, flags=0)
参数 pattern 为正则表达式, string 为待操作字符串, flags 为所用模式,函数作用为在待操作字符串中寻找所有匹配正则表达式的字串,返回一个列表,如果没有匹配到任何子串,返回一个空列表。
复制代码
s = ”’first line
second line
third line”’
regex = re.compile(“\w+”)
print regex.findall(s)
print re.findall(“\w+”, s)
复制代码
finditer(pattern, string, flags=0)
参数和作用与 findall 一样,不同之处在于 findall 返回一个列表, finditer 返回一个迭代器(参见 http://www.cnblogs.com/huxi/archive/2011/07/01/2095931.html ), 而且迭代器每次返回的值并不是字符串,而是一个 SRE_Match (参见 第四小节 re 内置对象用法) 对象,这个对象的具体用法见 match 函数。
复制代码
s = ”’first line
second line
third line”’
regex = re.compile(“\w+”)
print regex.finditer(s)
for i in regex.finditer(s):
print i
复制代码
match(pattern, string, flags=0)
使用指定正则去待操作字符串中寻找可以匹配的子串, 返回匹配上的第一个字串,并且不再继续找,需要注意的是 match 函数是从字符串开始处开始查找的,如果开始处不匹配,则不再继续寻找,返回值为 一个 SRE_Match (参见 第四小节 re 内置对象用法) 对象,找不到时返回 None
复制代码
s = ”’first line
second line
third line”’
regex = re.compile(“\w+”)
m = regex.match(s)
print m
print m.group()
regex = re.compile(“^i\w+”)
print regex.match(s)
复制代码
purge()
当你在程序中使用 re 模块,无论是先使用 compile 还是直接使用比如 findall 来使用正则表达式操作文本,re 模块都会将正则表达式先编译一下, 并且会将编译过后的正则表达式放到缓存中,这样下次使用同样的正则表达式的时候就不需要再次编译, 因为编译其实是很费时的,这样可以提升效率,而默认缓存的正则表达式的个数是 100, 当你需要频繁使用少量正则表达式的时候,缓存可以提升效率,而使用的正则表达式过多时,缓存带来的优势就不明显了 (参考 《python re.compile对性能的影响》http://blog.trytofix.com/article/detail/13/), 这个函数的作用是清除缓存中的正则表达式,可能在你需要优化占用内存的时候会用到。
search(pattern, string, flags=0)
函数类似于 match,不同之处在于不限制正则表达式的开始匹配位置
复制代码
s = ”’first line
second line
third line”’
print re.match(‘i\w+’, s)
print re.search(‘i\w+’, s)
print re.search(‘i\w+’, s).group()
复制代码
split(pattern, string, maxsplit=0, flags=0)
参数 maxsplit 指定切分次数, 函数使用给定正则表达式寻找切分字符串位置,返回包含切分后子串的列表,如果匹配不到,则返回包含原字符串的一个列表
复制代码
s = ”’first 111 line
second 222 line
third 333 line”’
print re.split(‘\d+’, s)
print re.split(‘.+’, s, 1)
print re.split(‘\d+’, s, 1)
复制代码
sub(pattern, repl, string, count=0, flags=0)
替换函数,将正则表达式 pattern 匹配到的字符串替换为 repl 指定的字符串, 参数 count 用于指定最大替换次数
复制代码
s = “the sum of 7 and 9 is [7+9].”
print re.sub(‘[7+9]’, ‘16’, s)
print re.sub(‘[(7)+(9)]’, r’\2\1’, s)
def replacement(m):
p_str = m.group()
if p_str == ‘7’:
return ‘77’
if p_str == ‘9’:
return ‘99’
return ”
print re.sub(‘\d’, replacement, s)
scope = {}
example_string_1 = “the sum of 7 and 9 is [7+9].”
example_string_2 = “[name = ‘Mr.Gumby’]Hello,[name]”
def replacement(m):
code = m.group(1)
st = ”
try:
st = str(eval(code, scope))
except SyntaxError:
exec code in scope
return st
print re.sub(‘[(.+?)]’, replacement, example_string_1)
print re.sub(‘[(.+?)]’, replacement, example_string_2)
复制代码
subn(pattern, repl, string, count=0, flags=0)
作用与函数 sub 一样, 唯一不同之处在于返回值为一个元组,第一个值为替换后的字符串,第二个值为发生替换的次数
template(pattern, flags=0)
这个吧,咋一看和 compile 差不多,不过不支持 +、?、*、{} 等这样的元字符,只要是需要有重复功能的元字符,就不支持,查了查资料,貌似没人知道这个函数到底是干嘛的…
四、re 内置对象用法
SRE_Pattern 这个对象是一个编译后的正则表达式,编译后不仅能够复用和提升效率,同时也能够获得一些其他的关于正则表达式的信息
属性:
flags 编译时指定的模式
groupindex 以正则表达式中有别名的组的别名为键、以该组对应的编号为值的字典,没有别名的组不包含在内。
groups 正则表达式中分组的数量
pattern 编译时用的正则表达式
复制代码
s = ‘Hello, Mr.Gumby : 2016/10/26’
p = re.compile(”’(?: # 构造一个不捕获分组 用于使用 |
(?P\w+.\w+) # 匹配 Mr.Gumby
| # 或
(?P\s+.\w+) # 一个匹配不到的命名分组
)
.*? # 匹配 :
(\d+) # 匹配 2016
”’, re.X)
#
print p.flags
print p.groupindex
print p.groups
print p.pattern
复制代码
函数:可使用 findall、finditer、match、search、split、sub、subn 等函数
SRE_Match 这个对象会保存本次匹配的结果,包含很多关于匹配过程以及匹配结果的信息
属性:
endpos 本次搜索结束位置索引
lastgroup 本次搜索匹配到的最后一个分组的别名
lastindex 本次搜索匹配到的最后一个分组的索引
pos 本次搜索开始位置索引
re 本次搜索使用的 SRE_Pattern 对象
regs 列表,元素为元组,包含本次搜索匹配到的所有分组的起止位置
string 本次搜索操作的字符串
复制代码
s = ‘Hello, Mr.Gumby : 2016/10/26’
m = re.search(‘, (?P\w+.\w+).*?(\d+)’, s)
print m.endpos
print m.lastgroup
print m.lastindex
print m.pos
print m.re
print m.regs
print m.string
复制代码
函数:
end([group=0]) 返回指定分组的结束位置,默认返回正则表达式所匹配到的最后一个字符的索引
expand(template) 根据模版返回相应的字符串,类似与 sub 函数里面的 repl, 可使用 \1 或者 \g 来选择分组
group([group1, …]) 根据提供的索引或名字返回响应分组的内容,默认返回 start() 到 end() 之间的字符串, 提供多个参数将返回一个元组
groupdict([default=None]) 返回 返回一个包含所有匹配到的命名分组的字典,没有命名的分组不包含在内,key 为组名, value 为匹配到的内容,参数 default 为没有参与本次匹配的命名分组提供默认值
groups([default=None]) 以元组形式返回每一个分组匹配到的字符串,包括没有参与匹配的分组,其值为 default
span([group]) 返回指定分组的起止位置组成的元组,默认返回由 start() 和 end() 组成的元组
start([group]) 返回指定分组的开始位置,默认返回正则表达式所匹配到的第一个字符的索引
复制代码
s = ‘Hello, Mr.Gumby : 2016/10/26’
m = re.search(”’(?: # 构造一个不捕获分组 用于使用 |
(?P\w+.\w+) # 匹配 Mr.Gumby
| # 或
(?P\s+.\w+) # 一个匹配不到的命名分组
)
.*? # 匹配 :
(\d+) # 匹配 2016
”’,
s, re.X)
print m.end()
print m.expand(“my name is \1”)
print m.group()
print m.group(1,2)
print m.groupdict(‘default_string’)
print m.groups(‘default_string’)
print m.span(3)
print m.start(3)
复制代码
五、分组用法
python 的正则表达式中用小括号 "(" 表示分组,按照每个分组中前半部分出现的顺序 "(" 判定分组的索引,索引从 1 开始,每个分组在访问的时候可以使用索引,也可以使用别名
复制代码
s = ‘Hello, Mr.Gumby : 2016/10/26’
p = re.compile(“(?P\w+.\w+).*?(\d+)(?#comment)”)
m = p.search(s)
print m.group(‘name’)
print m.group(2)
复制代码
有时候可能只是为了把正则表达式分组,而不需要捕获其中的内容,这时候可以使用非捕获分组
复制代码
s = ‘Hello, Mr.Gumby : 2016/10/26’
p = re.compile(“””
(?: # 非捕获分组标志 用于使用 |
(?P\w+.\w+)
|
(\d+/)
)
“”“, re.X)
m = p.search(s)
print p.groups
print m.groups()
复制代码
如果你在写正则的时候需要在正则里面重复书写某个表达式,那么你可以使用正则的引用分组功能,需要注意的是引用的不是前面分组的 正则表达式 而是捕获到的 内容,并且引用的分组不算在分组总数中.
复制代码
s = ‘Hello, Mr.Gumby : 2016/2016/26’
p = re.compile(“””
(?: # 非捕获分组标志 用于使用 |
(?P\w+.\w+)
|
(\d+/)
)
.*?(?P\d+)/(?P=number)/
“”“, re.X)
m = p.search(s)
print p.groups
print m.groups()
print m.group()
复制代码
六、环视用法
环视还有其他的名字,例如 界定、断言、预搜索等,叫法不一。
环视是一种特殊的正则语法,它匹配的不是字符串,而是 位置,其实就是使用正则来说明这个位置的左右应该是什么或者应该不是什么,然后去寻找这个位置。
环视的语法有四种,见第一小节元字符,基本用法如下。
复制代码
s = ‘Hello, Mr.Gumby : 2016/10/26 Hello,r.Gumby : 2016/10/26’
print re.compile(“(?P\w+.\w+)”).findall(s)
print re.compile(“(?<=Hello, )(?P\w+.\w+)”).findall(s)
print re.compile(“(?
print re.compile(“(?=M)(?P\w+.\w+)”).findall(s)
print re.compile(“(?!r)(?P\w+.\w+)”).findall(s)
复制代码
高级一些的例子参见《正则基础之——环视(Lookaround)》(http://www.cnblogs.com/kernel0815/p/3375249.html)