正则表达式
一,认识正则表达式
正则表达式: 解决字符串问题的工具(让复杂的字符串问题变得简单的一个工具)
二,匹配类符号
1. re模块
re模块是python用来支持正则表达式的一个模块
re模块中提供了各种和正则相关的函数:fullmatch、search、findall、match、split、sub等等
fullmatch(正则表达式, 字符串) - 判断整个字符串是否完全符号正则表达式描述的规则。如果不符合返回值是None
python中提供正则表达式的方式:r'正则表达式'
js中提供正则表达式的方式:/正则表达式/
2. 匹配类符号 - 一个正则符号表示一类字符
匹配类符号在正则中的作用:用来要求字符串中某个位置必须是什么样的字符
1)普通符号 - 在正则表达式中表示这个符号本身,对应字符串中的字符的要求就是符号本身。
2) . - 匹配一个任意字符
3)\d - 匹配一个任意数字
4)\s - 匹配任意一个空白字符(空白字符包括:空格、\n、\t)
5)\w - 匹配任意一个数字、字母或者下划线或者中文
6)\大写字母 - 与相应的小写字母的功能相反
7) [字符集] - 匹配字符集中任意一个字符
[多个普通符号] - 例如:[abc12], 在'a'、'b'、'c'、'1'、'2'五个符号中任意一个可以匹配
[包含\开头的特殊符号] - 例如:[mn\d]、[m\dn]、[\dmn], 要求是m或者n或者任意一个数字
[字符1-字符2] - 例如:[a-z],要求是任意一个小写字母
[a-zA-Z],要求是任意一个字母
[2-9a-z],要求是2到9或者是任意一个小写字母
[\u4e00-\u9fa5],要求是任意一个中文
[\u4e00-\u9fa5\dabc]
注意:[]中如果-不在两个字符之间,就不能表示谁到谁,那它就是个普通符号
8) ^字符集 - 匹配不在字符集中的任意一个字符
三,匹配次数
1. * - 匹配0次或者多次(任意次数)
a* - a出现任意次数
\d* - 任意多个\d -> 任意多个数字
[abc]* - 任意多个[abc] -> 任意多个(a或者b或者c)
2. + - 匹配1次或者多次(至少1次)
3. ? - 0次或1次
4. {}
{N} - N次
{M,N} - M到N次
{M,} - 至少M次
{,N} - 最多N次
* == {0,}
+ == {1,}
? == {0,1}
5. 贪婪和非贪婪
在匹配次数不确定的时候,匹配模式分为贪婪和非贪婪两种,默认是贪婪的。
匹配次数不确定:*、+、?、{M,N}、{M,}、{,N}
贪婪和非贪婪:在次数不确定的情况下,对应的字符串在不同次数下有多种匹配结果,贪婪取最多次数对应的结果。(前提是匹配成功有多种情况)
非贪婪取最少次数对应的结果。
贪婪:*、+、?、{M,N}、{M,}、{,N}
非贪婪:*?、+?、??、{M,N}?、{M,}?、{,N}?
四,分组和分支
1. 分组 - ()
分组就是在正则中用括号将正则中的部分内容括起来就形成了一个分组
1)整体操作
2) 重复
在正则中:\N可以重复\N所在的位置的前面的第N个分组匹配到的内容
3) 捕获 - 获取正则匹配结果中的部分内容
2. 分支 - |
正则1|正则2 - 先用正则1进行匹配如果匹配成功就直接成功,如果匹配失败再用正则2进行匹配,如果匹配成功就成功,如果失败就失败
五,检测类符号和转义符号
1. 检测类符号(了解)
检测类符号不是匹配符号,不会要求某个位置必须是什么样的字符,而是用检测某个位置是否符号相关要求
1)\b - 检测是否是单词边界
单词边界 - 凡是可以用来将两个单词区分开的符号,例如:空白字符、标点符号、字符串开头、字符串结尾
2)\B - 检测是否是非单词边界
3)^ - 检测是否是字符串开头([]外面)
4)$ - 检测是否是字符串结尾
2. 转义符号
正则中的转义符号是指在本身就具备特殊功能的符号前加\,让它本身具备的特殊功能消失变成一个普通符号。
补充:独立存在有特殊意义的符号,放到[]中特殊功能会直接消失变成一个普通符号,例如:+、*、.、?、)、(等
六,re模块
1. 常用的函数
1)re.fullmatch(正则, 字符串) - 用整个字符串和正则表达式进行匹配,如果匹配成功返回匹配对象,匹配失败返回None
2)re.match(正则, 字符串) - 匹配字符串开头,如果匹配成功返回匹配对象,匹配失败返回None
3)re.search(正则, 字符串) - 匹配字符串中第一个满足正则的字串,如果匹配成功返回匹配对象,匹配失败返回None
4)re.findall(正则, 字符串) - 获取字符串中所有满足正则的子串,返回值是列表,列表中的元素是匹配到的字符串
5)re.finditer(正则, 字符串) - 获取字符串中所有满足正则的子串,返回一个迭代器,迭代器中的元素是匹配对象
6)re.split(正则, 字符串) - 将字符串中所有满足正则的字串作为切割点对字符串进行切割,返回一个列表,列表中的元素是字符串
7)re.sub(正则, 字符串1, 字符串2) - 将字符串2中所有满足正则的字串全部替换成字符串1
2. 匹配对象
1)获取匹配结果对应的字符串
2)获取匹配结果在原字符串中的位置信息
3.参数
1)单行匹配和多行匹配
2)忽略大小写
方法:flags=re.I、(?i)
3)既忽略大小写又要单行匹配
方法:flags=re.I|re.S、(?si)
作业
利用正则表达式完成下面的操作:
一、不定项选择题
-
能够完全匹配字符串"(010)-62661617"和字符串"01062661617"的正则表达式包括(A,B,D )
A.
r"\(?\d{3}\)?-?\d{8}"
B.r"[0-9()-]+"
C.r"[0-9(-)]*\d*"
D.r"[(]?\d*[)-]*\d*"
-
能够完全匹配字符串"back"和"back-end"的正则表达式包括( A,B,C)
A.r'\w{4}-\w{3}|\w{4}'
B.r'\w{4}|\w{4}-\w{3}'
C.r'\S+-\S+|\S+'
D.r'\w*\b-\b\w*|\w*'
-
能够完全匹配字符串"go go"和"kitty kitty",但不能完全匹配“go kitty”的正则表达式包括(B,C,D)
A.r '\b(\w+)\b\s+\1\b'
B.r'\w{2,5}\s*\1'
C.r'(\S+) \s+\1'
D.r'(\S{2,5})\s{1,}\1'
-
能够在字符串中匹配"aab",而不能匹配"aaab"和"aaaab"的正则表达式包括(B,C )
A.r"a*?b"
B.r"a{,2}b"
C.r"aa??b"
D.r"aaa??b"
二、编程题
1.用户名匹配
要求: 1.用户名只能包含数字 字母 下划线
2.不能以数字开头
3.⻓度在 6 到 16 位范围内
from re import *
username = input('请输入用户名')
if fullmatch(r'[a-zA-Z_][1-9a-zA-Z_]{5,15}',username):
print('可以使用此用户名')
else :
print('不可以使用此用户名')
- 密码匹配
要求: 1.不能包含!@#¥%^&*这些特殊符号
2.必须以字母开头
3.⻓度在 6 到 12 位范围内
from re import *
password = input('请输入密码:')
if fullmatch(r'[a-zA-Z][^!@#¥%^&*]{5,11}',password):
print('密码可以使用')
else :
print('密码格式有误!')
- ipv4 格式的 ip 地址匹配
提示: IP地址的范围是 0.0.0.0 - 255.255.255.255
- 提取用户输入数据中的数值 (数值包括正负数 还包括整数和小数在内) 并求和
例如:“-3.14good87nice19bye” =====> -3.14 + 87 + 19 = 102.86
-
验证输入内容只能是汉字
from re import * a= input('请输入:') if fullmatch(r'[\u4e00-\u9fa5]+',a): print(True) else: print(False)
-
匹配整数或者小数(包括正数和负数)
-
验证输入用户名和QQ号是否有效并给出对应的提示信息
-
拆分长字符串:将一首诗的中的每一句话分别取出来