Python正则表达式的使用范例详解


Posted in Python onAugust 08, 2014

作为一个概念而言,正则表达式对于Python来说并不是独有的。但是,Python中的正则表达式在实际使用过程中还是有一些细小的差别。

本文是一系列关于Python正则表达式文章的其中一部分。在这个系列的第一篇文章中,我们将重点讨论如何使用Python中的正则表达式并突出Python中一些独有的特性。

我们将介绍Python中对字符串进行搜索和查找的一些方法。然后我们讲讨论如何使用分组来处理我们查找到的匹配对象的子项。

我们有兴趣使用的Python中正则表达式的模块通常叫做‘re'。

>>> import re

1. Python中的原始类型字符串

Python编译器用‘\'(反斜杠)来表示字符串常量中的转义字符。

如果反斜杠后面跟着一串编译器能够识别的特殊字符,那么整个转义序列将被替换成对应的特殊字符(例如,‘\n'将被编译器替换成换行符)。

但这给在Python中使用正则表达式带来了一个问题,因为在‘re'模块中也使用反斜杠来转义正则表达式中的特殊字符(比如*和+)。

这两种方式的混合意味着有时候你不得不转义转义字符本身(当特殊字符能同时被Python和正则表达式的编译器识别的时候),但在其他时候你不必这么做(如果特殊字符只能被Python编译器识别)。

与其将我们的心思放在去弄懂到底需要多少个反斜杠,我们可以使用原始字符串来替代。

原始类型字符串可以简单的通过在普通字符串的双引号前面加一个字符‘r'来创建。当一个字符串是原始类型时,Python编译器不会对其尝试做任何的替换。本质上来讲,你在告诉编译器完全不要去干涉你的字符串。

>>> string = 'This is a\nnormal string'
>>> rawString = r'and this is a\nraw string'
>>> print string

这是一个普通字符串

>>> print rawString
and this is a\nraw string

这是一个原始类型字符串。
在Python中使用正则表达式进行查找

‘re'模块提供了几个方法对输入的字符串进行确切的查询。我们将会要讨论的方法有:

re.match()
re.search()
re.findall()

每一个方法都接收一个正则表达式和一个待查找匹配的字符串。让我们更详细的查看这每一个方法从而弄明白他们是如何工作的以及他们各有什么不同。

2. 使用re.match查找 ? 匹配开始

让我们先来看一下match()方法。match()方法的工作方式是只有当被搜索字符串的开头匹配模式的时候它才能查找到匹配对象。

举个例子,对字符串‘dog cat dog'调用mathch()方法,查找模式‘dog'将会匹配:

>>> re.match(r'dog', 'dog cat dog')
<_sre.SRE_Match object at 0xb743e720<
>>> match = re.match(r'dog', 'dog cat dog')
>>> match.group(0)
'dog'

我们稍后将更多的讨论group()方法。现在,我们只需要知道我们用0作为它的参数调用了它,group()方法返回查找到的匹配的模式。

我还暂且略过了返回的SRE_Match对象,我们很快也将会讨论到它。

但是,如果我们对同一个字符串调用math()方法,查找模式‘cat',则不会找到匹配。

>>> re.match(r'cat', 'dog cat dog')
>>>

3. 使用re.search查找 ? 匹配任意位置

search()方法和match()类似,不过search()方法不会限制我们只从字符串的开头查找匹配,因此在我们的示例字符串中查找‘cat'会查找到一个匹配:

search(r'cat', 'dog cat dog')
>>> match.group(0)
'cat'

然而search()方法会在它查找到一个匹配项之后停止继续查找,因此在我们的示例字符串中用searc()方法查找‘dog'只找到其首次出现的位置。

>>> match = re.search(r'dog', 'dog cat dog')
>>> match.group(0)
'dog'

4. 使用 re.findall ? 所有匹配对象

目前为止在Python中我使用的最多的查找方法是findall()方法。当我们调用findall()方法,我们可以非常简单的得到一个所有匹配模式的列表,而不是得到match的对象(我们会在接下来更多的讨论match对象)。对我而言这更加简单。对示例字符串调用findall()方法我们得到:

['dog', 'dog']
>>> re.findall(r'cat', 'dog cat dog')
['cat']

5. 使用 match.start 和 match.end 方法

那么,先前search()和match()方法先前返回给我们的‘match'对象”到底是什么呢?

和只简单的返回字符串的匹配部分不同,search()和match()返回的“匹配对象”,实际上是一个关于匹配子串的包装类。

先前你看到我可以通过调用group()方法得到匹配的子串,(我们将在下一个部分看到,事实上匹配对象在处理分组问题时非常有用),但是匹配对象还包含了更多关于匹配子串的信息。

例如,match对象可以告诉我们匹配的内容在原始字符串中的开始和结束位置:

>>> match = re.search(r'dog', 'dog cat dog')
>>> match.start()
0
>>> match.end()
3

知道这些信息有时候非常有用。

6. 使用 mathch.group 通过数字分组

就像我之前提到的,匹配对象在处理分组时非常得心应手。

分组是对整个正则表达式的特定子串进行定位的能力。我们可以定义一个分组做为整个正则表达式的一部分,然后单独的对这部分对应匹配到的内容定位。

让我们来看一下它是怎么工作的:

>>> contactInfo = 'Doe, John: 555-1212'

我刚才创建的字符串类似一个从某人的地址本里取出来的一个片段。我们可以通过这样一个正则表达式来匹配这一行:

>>> re.search(r'\w+, \w+: \S+', contactInfo)
<_sre.SRE_Match object at 0xb74e1ad8<

通过用圆括号来(字符‘('和‘)')包围正则表达式的特定部分,我们可以对内容进行分组然后对这些子组做单独处理。

>>> match = re.search(r'(\w+), (\w+): (\S+)', contactInfo)

这些分组可以通过用分组对象的group()方法得到。它们可以通过其在正则表达式中从左到右出现的数字顺序来定位(从1开始):

>>> match.group(1)
'Doe'
>>> match.group(2)
'John'
>>> match.group(3)
'555-1212'

组的序数从1开始的原因是因为第0个组被预留来存放所有匹配对象(我们在之前学习match()方法和search()方法到时候看到过)。

>>> match.group(0)
'Doe, John: 555-1212'

7. 使用 match.group 通过别名来分组

有时候,特别是当一个正则表达式有很多分组的时候,通过组的出现次序来定位就会变的不现实。Python还允许你通过下面的语句来指定一个组名:

>>> match = re.search(r'(?P<last>\w+), (?P<first>\w+): (?P<phone>\S+)', contactInfo)

我们还是可以用group()方法获取分组的内容,但这时候我们要用我们所指定的组名而不是之前所使用的组的所在位数。

>>> match.group('last')
'Doe'
>>> match.group('first')
'John'
>>> match.group('phone')
'555-1212'

这大大加强了代码的明确性和可读性。你可以想像当正则表达式变得越来越复杂,去弄懂一个分组到捕获了什么内容将会变得越来越困难。给你的分组命名将明确的告诉了你和你的读者你的意图。

尽管findall()方法不返回分组对象,它也可以使用分组。类似的,findall()方法将返回一个元组的集合,其中每个元组中的第N个元素对应了正则表达式中的第N个分组。

>>> re.findall(r'(\w+), (\w+): (\S+)', contactInfo)
[('Doe', 'John', '555-1212')]

但是,给分组命名并不适用于findall()方法。

在本文中我们介绍了Python中使用正则表达式的一些基础。我们学习了原始字符串类型(还有它能帮你解决的在使用正则表达式中一些头痛的问题)。我们还学习了如何适使用match(), search(), and findall()方法进行基本的查询,以及如何使用分组来处理匹配对象的子组件。

和往常一样,如果想查看更多关于这个主题的内容,re模块的Python官方文档是一个非常好的资源。

在以后的文章中,我们将更深入的讨论Python中正则表达式的应用。我们将更加全面的学习匹配对象,学习如何使用它们在字符串中做替换,甚至使用它们从文本文件中去解析Python数据结构。

Python 相关文章推荐
flask中使用SQLAlchemy进行辅助开发的代码
Feb 10 Python
Python使用urllib模块的urlopen超时问题解决方法
Nov 08 Python
python以环状形式组合排列图片并输出的方法
Mar 17 Python
Python基于Tkinter实现的记事本实例
Jun 17 Python
Python中将字典转换为XML以及相关的命名空间解析
Oct 15 Python
用python实现简单EXCEL数据统计的实例
Jan 24 Python
python解析含有重复key的json方法
Jan 22 Python
Python 给定的经纬度标注在地图上的实现方法
Jul 05 Python
python tkinter GUI绘制,以及点击更新显示图片代码
Mar 14 Python
详解django使用include无法跳转的解决方法
Mar 19 Python
Python3爬虫里关于识别微博宫格验证码的知识点详解
Jul 30 Python
python 爬虫如何正确的使用cookie
Oct 27 Python
python脚本实现统计日志文件中的ip访问次数代码分享
Aug 06 #Python
Python写的Socks5协议代理服务器
Aug 06 #Python
Python写的PHPMyAdmin暴力破解工具代码
Aug 06 #Python
Python写的Discuz7.2版faq.php注入漏洞工具
Aug 06 #Python
Python常用列表数据结构小结
Aug 06 #Python
Python生成pdf文件的方法
Aug 04 #Python
Tornado Web服务器多进程启动的2个方法
Aug 04 #Python
You might like
php查找指定目录下指定大小文件的方法
2014/11/28 PHP
yii2框架中使用下拉菜单的自动搜索yii-widget-select2实例分析
2016/01/09 PHP
thinkPHP3.2.3结合Laypage实现的分页功能示例
2018/05/28 PHP
[原创]IE view-source 无法查看看源码 JavaScript看网页源码
2009/07/19 Javascript
JS复制内容到剪切板的实例代码(兼容IE与火狐)
2013/11/19 Javascript
js获取url参数代码实例分享(JS操作URL)
2013/12/13 Javascript
基于jquery ui的alert,confirm方案(支持换肤)
2015/04/03 Javascript
js编写的treeview使用方法
2016/11/11 Javascript
常用原生js自定义函数总结
2016/11/20 Javascript
详解Vue使用 vue-cli 搭建项目
2017/04/20 Javascript
详解JSON Web Token 入门教程
2018/07/30 Javascript
使用bootstrap实现下拉框搜索功能的实例讲解
2018/08/10 Javascript
微信小程序实现简单表格
2019/02/14 Javascript
[03:53]2016国际邀请赛中国区预选赛第三日TOP10精彩集锦
2016/06/29 DOTA
基于Django的ModelForm组件(详解)
2017/12/07 Python
Python实现的井字棋(Tic Tac Toe)游戏示例
2018/01/31 Python
Python版名片管理系统
2018/11/30 Python
对python修改xml文件的节点值方法详解
2018/12/24 Python
在python中利用opencv简单做图片比对的方法
2019/01/24 Python
centos6.5安装python3.7.1之后无法使用pip的解决方案
2019/02/14 Python
Python3标准库总结
2019/02/19 Python
Python简单处理坐标排序问题示例
2019/07/11 Python
基于Python爬取51cto博客页面信息过程解析
2020/08/25 Python
使用Python将xmind脑图转成excel用例的实现代码(一)
2020/10/12 Python
Python实现石头剪刀布游戏
2021/01/20 Python
销售心得体会
2014/01/02 职场文书
施工班组长岗位职责
2014/01/05 职场文书
小学教研工作制度
2014/01/15 职场文书
环保建议书
2014/03/12 职场文书
舞蹈兴趣小组活动总结
2014/07/07 职场文书
简易离婚协议书(范本)
2014/10/25 职场文书
2015元旦主持词开场白和结束语
2014/12/14 职场文书
艺术节开幕词
2015/01/28 职场文书
活动主持人开场白
2015/05/28 职场文书
go语言中切片与内存复制 memcpy 的实现操作
2021/04/27 Golang
关于vue中如何监听数组变化
2021/04/28 Vue.js