关于正则表达式的5个小贴士

正则表达式是一个非常强大的处理字符工具,但有时可读性很差、晦涩难懂,Jamie Zawinski 说道:

Some people, when confronted with a problem, think, “I know, I’ll use regular expressions.” Now they have two problems.

本来是一个问题,引入正则表达式之后就成了两个问题。其实并不是任何场景都需要正则表达式。在简单场景,能用字符串自己提供的方法解决问题就没必要用正则表达式,比如字符替换

>>> import re
>>> text = 'java is most popular language'

>>> re.sub(r'java', 'python', text)
'python is most popular language'

# good
>>> text.replace("java", "python")
'python is most popular language'

判断字符串是否以某字符开头

>>> re.match(r"^java", text)
<_sre.SRE_Match object at 0x000000000471D578>
>>> text.startwith("java")

# good
>>> text.startswith("java")
True

re.match() 与 re.search()

re.match 从字符串的起始位置匹配,如果没匹配成功就不再往后匹配,返回 None。而 search 虽然也是从起始位置开始匹配,但是如果在起始位置没有匹配,就继续往后匹配,直到匹配为止,如果匹配到字符串末尾都没有匹配则返回 None

>>> text = "java is most popular langauge"
>>> re.match("most", text) # 没匹配

# bad
>>> re.match(".*most", text)
<_sre.SRE_Match object at 0x0000000004CCD578>

# good
>>> re.search("most", text)
<_sre.SRE_Match object at 0x000000000471D578>

不分组的括号

我们知道正则表达式中括号可以用于分组提取,有时我们并不希望括号用于分组该怎么办,答案是使用 (?:),看一个例子,用正则表达式提取URL中的各个组成部分

rex = r'^(http[s]?)://([^/\s]+)([/\w\-\.]+[^#?\s]*)?(?:\?([^#]*))?(?:#(.*))?$'
print(re.match(rex, url).groups())
>>> ('http',
    'www.example.com',
    '/path/to/myfile.html',
    'key1=value1&key2=value2',
    'SomewhereInTheDocument')

上面虽然写了7对括号,但其实只有5个分组。下面是不使用 ?:,出现了 7 组数据

rex = r'^(http[s]?)://([^/\s]+)([/\w\-\.]+[^#?\s]*)?(\?([^#]*))?(#(.*))?$'
print(re.match(rex, url).groups())
>>>('http',
   'www.example.com',
   '/path/to/myfile.html',
   '?key1=value1&key2=value2',
   'key1=value1&key2=value2',
   '#SomewhereInTheDocument',
   'SomewhereInTheDocument')

贪婪匹配

正则表达式默认是贪婪匹配的,也就是说它会在满足匹配条件的情况下尽可能多的匹配字符,例如这里有一段话:

html = """<div><p>Today a quick article on a nic</p><p>Read more ...</p></div>"""

里面有两对<p>标签,如果你只想匹配第一对,使用

>>> re.search("<p>.*</p>", html)
>>> m = re.search("<p>.*</p>", html)
>>> m.group()
'<p>Today a quick article on a nic</p><p>Read more ...</p>'
>>>

<p>.*</p> 会从第一个<p>开始,匹配到最后一个</p>,如果要想尽可能少匹配则可以在元字符后面加 ?

>>> m = re.search("<p>.*?</p>", html)
>>> m.group()
'<p>Today a quick article on a nic</p>'

最后推荐学习正则表达式的3个学习资源

  • 第一个是《正则表达式30分钟入门教程》,公众号回复 “rex” 获取电子书

  • 第二个是《Python正则表达式指南》介绍了正则表达式的概念和re模块下面的常用操作,阅读地址:http://www.cnblogs.com/huxi/archive/2010/07/04/1771073.html 

  • 第三个是一个在线网站: regexr.com,可以在线调试、检测正则表达式,如果是写复杂的表达式,有时不可能一步写到位,你需要不断修正表达式,那么这个网站无疑是最合适的,它能很清晰的看出来正则表达式是如何匹配字符串的。

推荐阅读:


关注Python之禅,学点Python

再推荐一下极客时间APP,他们正在搞抽奖活动,点击左下角「阅读原文」抽奖

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值