当前位置: 首页 > news >正文

<学习笔记>从零开始自学Python-之-基础语法篇(十一)正则表达式re库

正则表达式是一个特殊的字符序列,它能帮助你方便的检查一个字符串是否与某种模式匹配。

re是Python的标准库,无需安装,直接导入使用。

re 模块使 Python 语言拥有全部的正则表达式功能。

1、re.match函数

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

函数语法

re.match(pattern, string, flags=0)

函数参数说明:

参数描述
pattern匹配的正则表达式
string要匹配的字符串。
flags标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等。

匹配成功re.match方法返回一个匹配的对象,否则返回None。

我们可以使用group(num) 或 groups() 匹配对象函数来获取匹配表达式。

匹配对象方法描述
group(num=0)匹配的整个表达式的字符串,group() 可以一次输入多个组号,在这种情况下它将返回一个包含那些组所对应值的元组。
groups()返回一个包含所有小组字符串的元组,从 1 到 所含的小组号。

例1:match要从头匹配

import re

 # 在起始位置匹配
print(re.match('大漠', '大漠孤烟直,长河落日圆').span()) 
print(re.match('落日', '大漠孤烟直,长河落日圆').span())
# 不在起始位置匹配
print(re.match('大漠', '大漠孤烟直,长河落日圆'))         
print(re.match('落日', '大漠孤烟直,长河落日圆'))



out:

(0, 2)
Traceback (most recent call last):
  File "C:/coding/aNewPy/正则表达式re模块.py", line 4, in <module>
    print(re.match('落日', '大漠孤烟直,长河落日圆').span())
AttributeError: 'NoneType' object has no attribute 'span'


<_sre.SRE_Match object; span=(0, 2), match='大漠'>
None


例2:group用法:

line = "大漠孤烟直,长河落日圆"
matchObj = re.match( '(.*)孤烟(.*).*', line)
if matchObj:
   print ("matchObj.group() : ", matchObj.group())
   print ("matchObj.group(0) : ", matchObj.group(0))
   print ("matchObj.group(1) : ", matchObj.group(1))
   print ("matchObj.group(2) : ", matchObj.group(2))
   
else:
   print ("No match!!")




out:

matchObj.group() :  大漠孤烟直,长河落日圆
matchObj.group(0) :  大漠孤烟直,长河落日圆
matchObj.group(1) :  大漠
matchObj.group(2) :  直,长河落日圆

group(0)和group()一样,是指符合条件的字符串

group(1)是指匹配项之前的字符串

group(2)是指匹配项之后的字符串

2、re.search方法

re.search 扫描整个字符串并返回第一个成功的匹配。

函数语法:

re.search(pattern, string, flags=0)

函数参数说明:

参数描述
pattern匹配的正则表达式
string要匹配的字符串。
flags标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等。

匹配成功re.search方法返回一个匹配的对象,否则返回None。

我们可以使用group(num) 或 groups() 匹配对象函数来获取匹配表达式。

匹配对象方法描述
group(num=0)匹配的整个表达式的字符串,group() 可以一次输入多个组号,在这种情况下它将返回一个包含那些组所对应值的元组。
groups()返回一个包含所有小组字符串的元组,从 1 到 所含的小组号。

例3:把例1用search匹配一下看看:

print(re.search('孤烟', '大漠孤烟直,长河落日圆').span())  # 在起始位置匹配
print(re.search('落日', '大漠孤烟直,长河落日圆').span())
print(re.search('孤烟', '大漠孤烟直,长河落日圆'))         # 不在起始位置匹配
print(re.search('落日', '大漠孤烟直,长河落日圆'))



out

(2, 4)
(8, 10)
<re.Match object; span=(2, 4), match='孤烟'>
<re.Match object; span=(8, 10), match='落日'>

例4:把例2用search方法试一下

line = "大漠孤烟直,长河落日圆"
matchObj = re.search( '(.*)孤烟(.*).*', line)
if matchObj:
   print ("matchObj.group() : ", matchObj.group())
   print ("matchObj.group(0) : ", matchObj.group(0))
   print ("matchObj.group(1) : ", matchObj.group(1))
   print ("matchObj.group(2) : ", matchObj.group(2))
   
else:
   print ("No match!!")



out

matchObj.group() :  大漠孤烟直,长河落日圆
matchObj.group(0) :  大漠孤烟直,长河落日圆
matchObj.group(1) :  大漠
matchObj.group(2) :  直,长河落日圆

3、检索和替换

Python 的re模块提供了re.sub用于替换字符串中的匹配项。

语法:

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

参数:

  • pattern : 正则中的模式字符串。
  • repl : 替换的字符串,也可为一个函数。
  • string : 要被查找替换的原始字符串。
  • count : 模式匹配后替换的最大次数,默认 0 表示替换所有的匹配。
  • flags : 编译时用的匹配模式,数字形式。

前三个为必选参数,后两个为可选参数。

例5:

phone = "139-2345-6789 # 这是我的手机号码"
 
# 删除注释
num = re.sub(r'#.*$', "", phone)
print ("我的手机号码 : ", num)
 
# 移除非数字的内容
num = re.sub(r'\D', "", phone)
print ("我的手机号码 : ", num)



out

我的手机号码 :  139-2345-6789
我的手机号码 :  13923456789

4、re.compile 函数

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

语法格式为:

re.compile(pattern[, flags])

参数:

  • pattern : 一个字符串形式的正则表达式
  • flags 可选,表示匹配模式,比如忽略大小写,多行模式等,具体参数为:
    • re.I 忽略大小写
    • re.L 表示特殊字符集 \w, \W, \b, \B, \s, \S 依赖于当前环境
    • re.M 多行模式
    • re.S 即为' . '并且包括换行符在内的任意字符(' . '不包括换行符)
    • re.U 表示特殊字符集 \w, \W, \b, \B, \d, \D, \s, \S 依赖于 Unicode 字符属性数据库
    • re.X 为了增加可读性,忽略空格和' # '后面的注释

例:6:用compile 结合 pattern 找手机号

msg = "张三,手机号123-4567-8901,住址:某某区某某街473号38栋407室"
pattern = re.compile(r'(?<=手机号)\d+-\d+-\d+')
m = pattern.match(msg, 6, 20)
m2=pattern.search(msg)
print(m)
print(m2)


out:

<re.Match object; span=(6, 19), match='123-4567-8901'>
<re.Match object; span=(6, 19), match='123-4567-8901'>

5、re.findall

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

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

语法格式为:

re.findall(pattern, string, flags=0)
或
pattern.findall(string[, pos[, endpos]])

参数:

  • pattern 匹配模式。
  • string 待匹配的字符串。
  • pos 可选参数,指定字符串的起始位置,默认为 0。
  • endpos 可选参数,指定字符串的结束位置,默认为字符串的长度。

例7:找出字符串中的所有数字

msg = "张三,手机号123-4567-8901,住址:某某区某某街473号38栋407室"
pattern = re.compile(r'\d+')   
result1 = pattern.findall(msg)
result2 = pattern.findall(msg, 6, 20)
 
print(result1)
print(result2)

out:

['123', '4567', '8901', '473', '38', '407']
['123', '4567', '8901']

例8:多个匹配模式,返回元组列表:

result = re.findall(r'(\w+)=(\d+)', 'set width=20 and height=10')
print(result)


out:
[('width', '20'), ('height', '10')]

6、re.finditer

和 findall 类似,在字符串中找到正则表达式所匹配的所有子串,并把它们作为一个迭代器返回。

re.finditer(pattern, string, flags=0)

参数:

参数描述
pattern匹配的正则表达式
string要匹配的字符串。
flags标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等。

例9:finditer示例

msg = "张三,手机号123-4567-8901,住址:某某区某某街473号38栋407室"
it = re.finditer(r"\d+",msg) 
for match in it: 
    print (match.group() )


out:
123
4567
8901
473
38
407

7、re.split

split 方法按照能够匹配的子串将字符串分割后返回列表,它的使用形式如下:

re.split(pattern, string[, maxsplit=0, flags=0])

参数:

参数描述
pattern匹配的正则表达式
string要匹配的字符串。
maxsplit分割次数,maxsplit=1 分割一次,默认为 0,不限制次数。
flags标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等。

例10:分割字符串中所有数字

msg = "张三,手机号123-4567-8901,住址:某某区某某街473号38栋407室"
s=re.split('\D+',msg)
print(s)


out:
['', '123', '4567', '8901', '473', '38', '407', '']

8、正则表达式对象

re.RegexObject

re.compile() 返回 RegexObject 对象。

re.MatchObject

group() 返回被 RE 匹配的字符串。

  • start() 返回匹配开始的位置
  • end() 返回匹配结束的位置
  • span() 返回一个元组包含匹配 (开始,结束) 的位置

9、正则表达式修饰符 - 可选标志

正则表达式可以包含一些可选标志修饰符来控制匹配的模式。修饰符被指定为一个可选的标志。多个标志可以通过按位 OR(|) 它们来指定。如 re.I | re.M 被设置成 I 和 M 标志:

修饰符描述
re.I使匹配对大小写不敏感
re.L做本地化识别(locale-aware)匹配
re.M多行匹配,影响 ^ 和 $
re.S使 . 匹配包括换行在内的所有字符
re.U根据Unicode字符集解析字符。这个标志影响 \w, \W, \b, \B.
re.X该标志通过给予你更灵活的格式以便你将正则表达式写得更易于理解。

10、正则表达式模式

模式字符串使用特殊的语法来表示一个正则表达式。

字母和数字表示他们自身。一个正则表达式模式中的字母和数字匹配同样的字符串。

多数字母和数字前加一个反斜杠时会拥有不同的含义。

标点符号只有被转义时才匹配自身,否则它们表示特殊的含义。

反斜杠本身需要使用反斜杠转义。

由于正则表达式通常都包含反斜杠,所以你最好使用原始字符串来表示它们。模式元素(如 r'\t',等价于 \\t )匹配相应的特殊字符。

下表列出了正则表达式模式语法中的特殊元素。如果你使用模式的同时提供了可选的标志参数,某些模式元素的含义会改变。

模式描述
^匹配字符串的开头
$匹配字符串的末尾。
.匹配任意字符,除了换行符,当re.DOTALL标记被指定时,则可以匹配包括换行符的任意字符。
[...]用来表示一组字符,单独列出:[amk] 匹配 'a','m'或'k'
[^...]不在[]中的字符:[^abc] 匹配除了a,b,c之外的字符。
re*匹配0个或多个的表达式。
re+匹配1个或多个的表达式。
re?匹配0个或1个由前面的正则表达式定义的片段,非贪婪方式
re{ n}匹配n个前面表达式。例如,"o{2}"不能匹配"Bob"中的"o",但是能匹配"food"中的两个o。
re{ n,}精确匹配n个前面表达式。例如,"o{2,}"不能匹配"Bob"中的"o",但能匹配"foooood"中的所有o。"o{1,}"等价于"o+"。"o{0,}"则等价于"o*"。
re{ n, m}匹配 n 到 m 次由前面的正则表达式定义的片段,贪婪方式
a| b匹配a或b
(re)匹配括号内的表达式,也表示一个组
(?imx)正则表达式包含三种可选标志:i, m, 或 x 。只影响括号中的区域。
(?-imx)正则表达式关闭 i, m, 或 x 可选标志。只影响括号中的区域。
(?: re)类似 (...), 但是不表示一个组
(?imx: re)在括号中使用i, m, 或 x 可选标志
(?-imx: re)在括号中不使用i, m, 或 x 可选标志
(?#...)注释.
(?= re)前向肯定界定符。如果所含正则表达式,以 ... 表示,在当前位置成功匹配时成功,否则失败。但一旦所含表达式已经尝试,匹配引擎根本没有提高;模式的剩余部分还要尝试界定符的右边。
(?! re)前向否定界定符。与肯定界定符相反;当所含表达式不能在字符串当前位置匹配时成功。
(?> re)匹配的独立模式,省去回溯。
\w匹配数字字母下划线
\W匹配非数字字母下划线
\s匹配任意空白字符,等价于 [\t\n\r\f]。
\S匹配任意非空字符
\d匹配任意数字,等价于 [0-9]。
\D匹配任意非数字
\A匹配字符串开始
\Z匹配字符串结束,如果是存在换行,只匹配到换行前的结束字符串。
\z匹配字符串结束
\G匹配最后匹配完成的位置。
\b匹配一个单词边界,也就是指单词和空格间的位置。例如, 'er\b' 可以匹配"never" 中的 'er',但不能匹配 "verb" 中的 'er'。
\B匹配非单词边界。'er\B' 能匹配 "verb" 中的 'er',但不能匹配 "never" 中的 'er'。
\n, \t, 等。匹配一个换行符。匹配一个制表符, 等
\1...\9匹配第n个分组的内容。
\10匹配第n个分组的内容,如果它经匹配。否则指的是八进制字符码的表达式。

11、正则表达式实例

字符匹配

实例描述
python匹配 "python".

字符类

实例描述
[Pp]ython匹配 "Python" 或 "python"
rub[ye]匹配 "ruby" 或 "rube"
[aeiou]匹配中括号内的任意一个字母
[0-9]匹配任何数字。类似于 [0123456789]
[a-z]匹配任何小写字母
[A-Z]匹配任何大写字母
[a-zA-Z0-9]匹配任何字母及数字
[^aeiou]除了aeiou字母以外的所有字符
[^0-9]匹配除了数字外的字符

特殊字符类

实例描述
.匹配除 "\n" 之外的任何单个字符。要匹配包括 '\n' 在内的任何字符,请使用象 '[.\n]' 的模式。
\d匹配一个数字字符。等价于 [0-9]。
\D匹配一个非数字字符。等价于 [^0-9]。
\s匹配任何空白字符,包括空格、制表符、换页符等等。等价于 [ \f\n\r\t\v]。
\S匹配任何非空白字符。等价于 [^ \f\n\r\t\v]。
\w匹配包括下划线的任何单词字符。等价于'[A-Za-z0-9_]'。
\W匹配任何非单词字符。等价于 '[^A-Za-z0-9_]'。

相关文章:

  • 随想录一期 day2 [977.有序数组的平方|209. 长度最小的子数组|59.螺旋矩阵II(剥洋葱)]
  • 自动驾驶数据标注基本框架,你了解多少?丨曼孚科技
  • 前景理论-风险决策分析的思维模型
  • Intel汇编-CMOV条件传送指令
  • 基于java宠物交易计算机毕业设计源码+系统+lw文档+mysql数据库+调试部署
  • 单片机毕设选题 - 便携式空气质量检测系统(物联网 嵌入式)
  • gsnark中的证明方案及曲线
  • LL(1)文法定义及判别
  • # Pytorch 中可以直接调用的Loss Functions总结:
  • 02-C语言经典算法100例
  • 〖Python 数据库开发实战 - Python与Redis交互篇③〗- 利用 redis-py 实现列表数据类型的常用指令操作
  • LwIP学习笔记1 - LwIP的设计目的、分层设计思想及模块划分概览
  • 使用 Convex 进行状态管理的指南
  • 卷积神经网络参数解读
  • IP 地址及其应用(计算机网络)
  • 分享一款快速APP功能测试工具
  • 【翻译】Mashape是如何管理15000个API和微服务的(三)
  • 【划重点】MySQL技术内幕:InnoDB存储引擎
  • Angular6错误 Service: No provider for Renderer2
  • Fastjson的基本使用方法大全
  • laravel 用artisan创建自己的模板
  • Laravel深入学习6 - 应用体系结构:解耦事件处理器
  • PhantomJS 安装
  • PHP CLI应用的调试原理
  • Python打包系统简单入门
  • spring boot 整合mybatis 无法输出sql的问题
  • 精益 React 学习指南 (Lean React)- 1.5 React 与 DOM
  • 前端技术周刊 2019-01-14:客户端存储
  • 驱动程序原理
  • 如何在 Tornado 中实现 Middleware
  • 数据科学 第 3 章 11 字符串处理
  • Redis4.x新特性 -- 萌萌的MEMORY DOCTOR
  • # 安徽锐锋科技IDMS系统简介
  • #if #elif #endif
  • (1)Nginx简介和安装教程
  • (附源码)计算机毕业设计ssm基于B_S的汽车售后服务管理系统
  • (三)elasticsearch 源码之启动流程分析
  • (原创)boost.property_tree解析xml的帮助类以及中文解析问题的解决
  • (转)LINQ之路
  • .helper勒索病毒的最新威胁:如何恢复您的数据?
  • .NET 5种线程安全集合
  • .Net Core与存储过程(一)
  • .net FrameWork简介,数组,枚举
  • .net wcf memory gates checking failed
  • .NET 常见的偏门问题
  • .NET 设计一套高性能的弱事件机制
  • .NET 事件模型教程(二)
  • .Net6支持的操作系统版本(.net8已来,你还在用.netframework4.5吗)
  • .NET8.0 AOT 经验分享 FreeSql/FreeRedis/FreeScheduler 均已通过测试
  • .NET设计模式(8):适配器模式(Adapter Pattern)
  • [ 蓝桥杯Web真题 ]-Markdown 文档解析
  • [ 隧道技术 ] cpolar 工具详解之将内网端口映射到公网
  • [④ADRV902x]: Digital Filter Configuration(发射端)
  • [Android学习笔记]ScrollView的使用
  • [go] 迭代器模式