python学习笔记(一)——关于正则表达式的学习小结
python中提供了re这个模块提供对正则表达式的支持。
一、正则表达式常用到的一些语法(并非全部):
| . | 匹配任意单个字符 |
| [...] | 匹配单个字符集 |
| \w | 匹配单词字符,即[a-zA-Z0-9] |
| \W | 匹配非单词字符集,例如 ‘*’ |
| \d | 匹配数字,即[0-9] |
| \D | 匹配非数字 |
| \s | 匹配空白字符 |
| \S | 匹配非空白字符 |
| * | 匹配前一个字符0次或者任意多次 |
| + | 匹配前一个字符1次或者任意多次 |
| ? | 匹配前一个字符0次或者1次 |
| {m} | 匹配前一个字符m次 |
| {m,n} | 匹配前一个字符最少m次,最多n次 |
| *? | 非贪婪模式匹配前一个字符0次或者任意多次 |
| +? | 非贪婪模式匹配前一个字符1次或者任意多次 |
| ?? | 非贪婪模式匹配前一个字符0次或者1次 |
| {m,n}? | 非贪婪模式匹配前一个字符最少m次,最多n次 |
| ^ | 匹配字符串开头 |
| $ | 匹配字符串结尾 |
| \A | 制定的字符串匹配必须出现在开头 |
| \Z | 制定的字符串匹配必须出现在结尾 |
| | | 匹配左右任意一个表达式,相当于“或”的含义 |
| () | 匹配一个分组,括号中为该分组所需匹配的内容 |
| \<number> | 引用匹配编号为<number>的分组中的字符串 |
| (?P<group_name>) | 为匹配分组制定特定的组名 |
| (?P=<group_name>) | 引用特定组名的匹配字符串 |
几点解释:
1. 两种方式都可以进行匹配:
(1)首先创建pattern,然后进行match
pa = re.compile(r'[\w]{6}') # 首先利用re模块创建一个pattern实例pa
ma = pa.match('string') # 利用这一pattern对正则表达式进行匹配
ma.group() #打印匹配的内容,输出为'string'
(2)直接利用re模块中的函数match进行匹配
ma = re.match(r'[\w]{6}', ‘string’)
print(ma.group()) # 打印匹配的内容,输出为'string'
2. 贪婪模式和非贪婪模式:
贪婪模式:总是尝试匹配尽可能多的字符;
非贪婪模式:总是尝试匹配尽可能少的字符。
例如:利用正则表达式‘python*’匹配‘pythonnnnnpython’,此时ma = re.match(r'python*',‘pythonnnpython’),得到的ma.group()为‘pythonnn’;利用‘python*?’进行匹配,此时ma = re.match(r'python*?',‘pythonnnpython’),得到的ma.group()为‘python’。
3. 关于逻辑与分组语法的用法:
| | | 匹配左右任意一个表达式,相当于“或”的含义 |
| () | 匹配一个分组,括号中为该分组所需匹配的内容 |
| \<number> | 引用匹配编号为<number>的分组中的字符串 |
| (?P<group_name>) | 为匹配分组制定特定的组名 |
| (?P=<group_name>) | 引用特定组名的匹配字符串 |
首先,|和()的用法比较容易理解,例如我们需要匹配多个邮箱的地址是否合法,例如有gmail邮箱、outlook邮箱,假定@前的字符数为6到20个,此时可以写成:
>>> pa = re.compile('[\w]{6,20}@(gmail|outlook).com$')
>>> ma = pa.match('bokeyuan@gmail.com')
>>> ma.group()
'bokeyuan@gmail.com'
>>> ma = pa.match('bokeyuan@outlook.com')
>>> ma.group()
'bokeyuan@outlookl.com'
后边三个的用法根据下边的例子进行说明:
>>> str = '<code>python</code>'
>>> ma = re.match(r'<[\w]+>', str)
>>> ma.group()
'<code>'
>>> ma = re.match(r'<([\w]+>)[\w]+</\1', str)
>>> ma.group()
'<code>python</code>'
>>> ma = re.match(r'<(?P<group1>[\w]+>)[\w]+</(?P=group1)', str)
>>> ma.group()
'<code>python</code>'
其中,第一个例子就是对字符串‘<code>'进行匹配。我们发现str中其实有两部分是完全相同的,就是都含有'code>'这个substring,于是可以看第二个例子,我们用()将([\w]+>)这部分内容括住时,这部分匹配的字符串就是'code>',([\w]+>)就是一个分组,没有起名字的情况下默认<number>为1,因此在我们需要在末尾再次引用到它的时候,就写上 /1 即可。第三个例子与第二个例子的效果完全相同,只不过为了更加清楚的记住匹配分组的名字,我们利用(?P<group_name>)这一语法功能,人为的为这个分组取了一个group1的名字,在最后又引用了这一分组。
4. 注意字符串中转义字符的问题
上述例子中,出现在正则表达式前边的r是原始字符串操作符,可以写为r或者R:表示字符串内的所有字符都按原始意思解释。
例如:‘c:\python\test.py’ 如果不加r,则计算机会将 \t 会变成转义字符解释;加上r以后,写为:r‘c:\python\test.py’,计算机就会直接输出c:\python\test.py,否则要想输出c:\python\test.py,必须将字符串写为‘c:\\python\\test.py’
二、介绍几个re模块中的常用函数
1. search(pattern, string, flags=0)函数
search函数功能:在字符串中查找匹配
例如:博客会记录来访者的数量,我们通过正则匹配查找字符串中的数字:
str1 = 'number of visitors = 1000'
info = re.search(r'\d+', str1) # 匹配字符串str1中的数字
print('访客数量:', info)
print(info.group()) # 显示匹配的内容
当然,上述操作也可以通过对字符串直接操作获得,例如:print('访客数量:', str1.find('1000')) 。但是这样存在一个问题,因为访客数量实在不断变化的,一旦1000这个数字增加,用字符串操作就难以实现。但是利用search函数就不存在这样的问题。
2. findall(pattern, string, flags=0)函数
findall函数功能:找到匹配,并返回所有匹配内容的列表
例如,博客记录了最近三天每天的访客记录,我们需要将三天的访客数量都查找出来,并计算总的访客数量,此时用search函数无法直接将三天的访客数量同时提取,可以采用findall函数:
str2 = 'day1=22, day2=34, day3=13'
info = re.findall(r'=[\d]+', str2)
print(info)
print('三天的访客数量为:', sum([int(x[1:]) for x in info]))
3. sub(pattern, repl, string, count=0, flags=0)函数
sub函数功能:将字符串中匹配正则表达式的部分替换为其他值。其中repl可以是一个字符串,也可以是一个函数。
当repl是一个字符串时,仍然以访客记录为例,当增加一个访客时,需要修改记录中的数字:
str3 = 'number of visitors = 1000'
info = re.sub(r'[\d]+', '', str3)
print(info) # 此时输出结果为 number of visitors = 1001
但是此时每次修改都必须手动输入数字,显然,对于这种随时变化的数字来说,这种操作是不合理的。而sub函数的高明之处就是允许repl是一个函数。
当repl为一个函数时,首先会在string中查找pattern的匹配,查找到的匹配是一个match对象,这个match对象就会被传递到repl这个函数中。上述例子可以这样实现:
# 首先,定义一个增加访客数量的函数,函数的参数是一个match对象
def add_num(match):
val = match.group()
num = int(val) + 1
return str(num) str3 = 'number of visitors = 1000'
print('最新访客数量:', re.sub(r'[\d]+', add_num, str3)) # 打印结果,最新访客数量:number of visitors = 1001
4. split(pattern, string, maxsplit=0, flags=0)函数
split函数功能:根据匹配分割字符串,返回分割字符串组成的列表
str4 = 'day1=22, day2=34, day3=13'
print(re.split(r', ', str4)) # 打印出的内容为 ['day1=22', 'day2=34', 'day3=13']
以上就是正则表达式的一些学习笔记,并不十分完全,欢迎大家交流。
python学习笔记(一)——关于正则表达式的学习小结的更多相关文章
- 【学习笔记】JavaScript的基础学习
[学习笔记]JavaScript的基础学习 一 变量 1 变量命名规则 Camel 标记法 首字母是小写的,接下来的字母都以大写字符开头.例如: var myTestValue = 0, mySeco ...
- 《Java学习笔记(第8版)》学习指导
<Java学习笔记(第8版)>学习指导 目录 图书简况 学习指导 第一章 Java平台概论 第二章 从JDK到IDE 第三章 基础语法 第四章 认识对象 第五章 对象封装 第六章 继承与多 ...
- 20145213《Java程序设计学习笔记》第六周学习总结
20145213<Java程序设计学习笔记>第六周学习总结 说在前面的话 上篇博客中娄老师指出我因为数据结构基础薄弱,才导致对第九章内容浅尝遏止地认知.在这里我还要自我批评一下,其实我事后 ...
- Hadoop学习笔记(10) ——搭建源码学习环境
Hadoop学习笔记(10) ——搭建源码学习环境 上一章中,我们对整个hadoop的目录及源码目录有了一个初步的了解,接下来计划深入学习一下这头神象作品了.但是看代码用什么,难不成gedit?,单步 ...
- 20145230《java学习笔记》第七周学习总结
20145230 <Java程序设计>第7周学习总结 教材学习内容 Lambda语法概览 我们在许多地方都会有按字符串长度排序的需求,如果在同一个方法内,我们可以使用一个byName局部变 ...
- 【学习笔记】jQuery的基础学习
[学习笔记]jQuery的基础学习 新建 模板 小书匠 什么是jQuery对象? jQuery 对象就是通过jQuery包装DOM对象后产生的对象.jQuery 对象是 jQuery 独有的. 如果 ...
- 吴裕雄--天生自然python学习笔记:Python3 正则表达式
Python 自1.5版本起增加了re 模块,它提供 Perl 风格的正则表达式模式. re 模块使 Python 语言拥有全部的正则表达式功能. compile 函数根据一个模式字符串和可选的标志参 ...
- Python机器学习笔记:sklearn库的学习
网上有很多关于sklearn的学习教程,大部分都是简单的讲清楚某一方面,其实最好的教程就是官方文档. 官方文档地址:https://scikit-learn.org/stable/ (可是官方文档非常 ...
- 学习笔记TF045:人工智能、深度学习、TensorFlow、比赛、公司
人工智能,用计算机实现人类智能.机器通过大量训练数据训练,程序不断自我学习.修正训练模型.模型本质,一堆参数,描述业务特点.机器学习和深度学习(结合深度神经网络). 传统计算机器下棋,贪婪算法,Alp ...
- .NET学习笔记(1)— C#学习路线图
目录 一:引言 二:.NET技术体系 三:常用工具汇总 四:学习资源汇总 五:书籍推荐 六:关于阅读技术书籍的经验 七:总结 一:引言 因为工作调整,从PHP开发零基础转型到.NET开发,前期没有 ...
随机推荐
- 洛谷【P1601】A+B Problem(高精)
题目传送门:https://www.luogu.org/problemnew/show/P1601 高精度加法板子.我们灵性地回忆一波小学学加法列竖式的场景(从\(6\)岁开始口算从未打过草稿的大佬请 ...
- python文件操作 seek(),tell()
seek():移动文件读取指针到指定位置 tell():返回文件读取指针的位置 seek()的三种模式: (1)f.seek(p,0) 移动当文件第p个字节处,绝对位置 (2)f.seek(p,1) ...
- 【转】 Pro Android学习笔记(四四):Dialog(1):触发Dialog
目录(?)[-] 创建dialog fragment Activity显示对话框 Android提供alert.prompt.pick-list,单选.多选,progress.time-picker和 ...
- js右击事件
先贴代码: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF ...
- Servlet的生命周期以及简单工作原理的讲解
Servlet生命周期分为三个阶段: 1,初始化阶段 调用init()方法 2,响应客户请求阶段 调用service()方法 3,终止阶段 调用destr ...
- Cocos2d-x使用Luajit将Lua脚本编译为bytecode,从而实现加密
转自:http://www.58player.com/blog-2537-87218.html 项目要求对lua脚本进行加密,查了一下相关的资料 ,得知lua本身可以使用luac将脚本编译为字节码(b ...
- UML核心元素--参与者
定义:参与者是在系统之外与系统交互的某人或某事物.参与者在建模过程中处于核心地位. 1.系统之外:系统之外的定义说明在参与者和系统之间存在明确的边界,参与者只能存在于边界之外,边界之内的所有人和事务都 ...
- AJAX经常遇到的那些问题
本文主要介绍了AJAX工作原理以及在面试题经常会遇到的问题,目录如下: 什么是Ajax 为什么要使用Ajax? Ajax特点? AJAX优缺点? Ajax流程? XMLhttprequest对象 AJ ...
- NodeJS”热部署“代码,实现动态调试(hotnode,可以实现热更新)
NodeJS”热部署“代码,实现动态调试 开发中遇到的问题 如果你有 PHP 开发经验,会习惯在修改 PHP 脚本后直接刷新浏览器以观察结果,而你在开发 Node.js 实现的 HTTP 应用时会 ...
- [凸包]Triangles
https://nanti.jisuanke.com/t/15429 题目大意:给出平面内$n$个整数坐标点,保证无三点共线.可以进行若干次连线,每次选择一个点对连接线段,但是任意两条线段都不得在给定 ...