title: 正则表达式re.sub替换不完整的问题现象及其根本原因
toc: true
comment: true
date: 2018-08-27 21:48:22
tags: ["Python", "正则表达式"]
category: ["Python"]
---
问题描述
问题的起因来自于一段正则替换。为了从一段HTML代码里面提取出正文,去掉所有的HTML标签和属性,可以写一个Python函数:
import re def remove_tag(html): text = re.sub('<.*"zzz">和</xxx>
。第二个参数表示被匹配到的内容将要被替换成什么内容。由于我需要提取正文,那么只要把所有HTML标签都替换为空字符串即可。第三个参数就是需要被替换的文本,在这个例子中是HTML源代码段。
至于
re.S
,在4年前的一篇文章中我讲到了它的用法:https://www.jb51.net/article/146384.htm现在使用一段HTML代码来测试一下:
import re def remove_tag(html): text = re.sub('<.*"content">今天的主角是<a href="xxx">kingname</a>,我们掌声欢迎!</div> ''' text = remove_tag(source_1) print(text)运行效果如下图所示,功能完全符合预期
再来测试一下代码中有换行符的情况:
import re def remove_tag(html): text = re.sub('<.*"content"> 今天的主角是 <a href="xxx">kingname</a> ,我们掌声欢迎! </div> ''' text = remove_tag(source_2) print(text)运行效果如下图所示,完全符合预期。
经过测试,在绝大多数情况下,能够从的HTML代码段中提取出正文。但也有例外。
例外情况
有一段HTML代码段比较长,内容如下:
<img> </span><span>遇见kingname</span></a ><a ><span class='url-icon'>< img '></span><span >温柔</span></a ><a ><span >#青南#</span></a > <br />就在这里…<br />我的小侯爷呢???运行效果如下图所示,最后两个HTML标签替换失败。
一开始我以为是HTML里面的空格或者引号引起的问题,于是我把HTML代码进行简化:
<img></span><span>遇见kingname</span></a><a><span><img></span><span>温柔</span></a><a><span>#青南#</span></a><br/>就在这里…<br/>我的小侯爷呢问题依然存在,如下图所示。
而且更令人惊讶的是,如果把第一个标签
<img>
删了,那么替换结果里面就少了一个标签,如下图所示。实际上,不仅仅是删除第一个标签,前面任意一个标签删了都可以减少结果里面的一个标签。如果删除前面两个或以上标签,那么结果就正常了。
答疑解惑
这个看起来很奇怪的问题,根本原因在re.sub的第4个参数。从函数原型可以看到:
def sub(pattern, repl, string, count=0, flags=0)第四个参数是count表示替换个数,re.S如果要用,应该作为第五个参数。所以如果把
remove_tag
函数做一些修改,那么结果就正确了:def remove_tag(html): text = re.sub('<.*"htmlcode">> import re > print(int(re.S)) 16现在回头数一数出问题的HTML代码,发现最后多出来的两个
<br>
标签,刚刚好是第17和18个标签,而由于count
填写的re.S
可以当做16来处理,那么Python就会把前16个标签替换为空字符串,从而留下最后两个。至此问题的原因搞清楚了。
这个问题没有被及早发现,有以下几个原因:
被替换的HTML代码是代码段,大多数情况下HTML标签不足16个,所以问题被隐藏。
re.S
是一个对象,但也是数字,count接收的参数刚好也是数字。在很多编程语言里面,常量都会使用数字,然后用一个有意义的大写字母来表示。re.S
处理的情况是<div \n>
而不是<div>\n</div>
但测试的代码段标签都是第二种情况,所以在代码段里面实际上加不加re.S
效果是一样的。补充:下面在给大家介绍下正则表达式 re.sub()替换功能
re.sub()替换功能
re.sub是个正则表达式方面的函数,用来实现通过正则表达式,实现比普通字符串的replace更加强大的替换功能。简单的替换功能可以使用replace()实现。
def main(): text = '123, word!' text1 = text.replace('123', 'Hello') print(text1) if __name__ == '__main__': main() # Hello, wold!如果通过re.sub(0函数则可以匹配任意的数字,并将其替换:
import re def main(): content = 'abc124hello46goodbye67shit' list1 = re.findall(r'\d+', content) print(list1) mylist = list(map(int, list1)) print(mylist) print(sum(mylist)) print(re.sub(r'\d+[hg]', 'foo1', content)) print() print(re.sub(r'\d+', '456654', content)) if __name__ == '__main__': main() # ['124', '46', '67'] # [124, 46, 67] # 237 # abcfoo1ellofoo1oodbye67shit # abc456654hello456654goodbye456654shit总结
以上所述是小编给大家介绍的正则表达式re.sub替换不完整的问题及完整解决方案,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
更新日志
- 凤飞飞《我们的主题曲》飞跃制作[正版原抓WAV+CUE]
- 刘嘉亮《亮情歌2》[WAV+CUE][1G]
- 红馆40·谭咏麟《歌者恋歌浓情30年演唱会》3CD[低速原抓WAV+CUE][1.8G]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[320K/MP3][193.25MB]
- 【轻音乐】曼托凡尼乐团《精选辑》2CD.1998[FLAC+CUE整轨]
- 邝美云《心中有爱》1989年香港DMIJP版1MTO东芝首版[WAV+CUE]
- 群星《情叹-发烧女声DSD》天籁女声发烧碟[WAV+CUE]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[FLAC/分轨][748.03MB]
- 理想混蛋《Origin Sessions》[320K/MP3][37.47MB]
- 公馆青少年《我其实一点都不酷》[320K/MP3][78.78MB]
- 群星《情叹-发烧男声DSD》最值得珍藏的完美男声[WAV+CUE]
- 群星《国韵飘香·贵妃醉酒HQCD黑胶王》2CD[WAV]
- 卫兰《DAUGHTER》【低速原抓WAV+CUE】
- 公馆青少年《我其实一点都不酷》[FLAC/分轨][398.22MB]
- ZWEI《迟暮的花 (Explicit)》[320K/MP3][57.16MB]