接下来,我们将学习如何使用jieba模块来实现古典名著《西游记》的分词,并且会将书中重点人物出场次数以图形化的方式显示出来,并进一步创建一个词云图。

17.4.1 读取文件

因为小说《西游记》的内容非常长,我们不太可能会把它放到一个字符串中来操作,所以我们需要它保存在一个文件中。那么我们就需要操作整个文件,把文件中的内容读取出来。我们操作文件的流程是:

1.打开文件,得到文件句柄并赋值给一个变量;

2.通过句柄对文件进行操作;

3.关闭文件。

打开文件就要用到open()函数。其实,我们在第11章中详细介绍了打开文件的方法,读者如果忘了的话,可以再次查阅11.1节回顾一下。

17.4.2 《西游记》的分词

在前面,我们已经介绍了如何使用jieba库分词,以及如何打开一个文本文件。接下来,我们要对经典小说《西游记》进行分词,并且把出现频率最高的词语展示出来。首先,把《西游记》保存到一个文本文件中,要注意的是,保存文件的时候,要将编码格式选择为UTF-8,否则读取文件的时候会报错。此外,我们要把这个文本文件放到和程序代码所在位置相同的文件夹中,这样就不需要指定路径了。

读者可以从本书配套源代码和文件中获取整个文本文件,如图17-5所示。

图17-5

我们先来看一下用于分词的程序代码,代码参见ch17\17.2.py。

import jiebadef takeSecond(elem):return elem[1]def main():path = "西游记.txt"file = open(path,"r",encoding="utf-8")text=file.read()file.close()words = jieba.lcut(text)counts = {}for word in words:counts[word] = counts.get(word,0) + 1items = list(counts.items())items.sort(key = takeSecond,reverse=True)    for i in range(20):item=items[i]keyWord =item[0]count=item[1]print("{0:<10}{1:>5}".format(keyWord,count))main()

因为要使用jieba的函数,所以这里首先需要导入jieba模块。

import jieba

接下来,我们定义了两个函数:main(主体函数)和takeSecond(用于获取列表的第2个元素)。然后,定义了变量path来保存相对路径。使用open()函数以只读方式打开文本文件“西游记.txt”,指定的编码方式是UTF-8,并且将文件句柄赋值给变量file。随后调用read()方法读取文件中的内容并保存到变量text中。调用close()方法关闭文件。

然后我们使用jieba.lcut()方法对变量text中的内容进行分词,并且把分词的结果列表保存到变量words中。我们新建一个叫作counts的字典。然后,通过一个循环语句,遍历列表words中的每个元素,用变量word来表示每个元素。在循环中,把word作为字典counts的键,把get()方法返回的值加上1,作为这个键所对应的值。这表示每次遇到同样的键,都会让它的值加上1(以统计相同的键的数目)。需要注意,如果在字典中没有找到键所对应的值,那么get()方法会返回默认值0。当循环结束后,字典counts就包含了西游记中拆分出来的全部词语以及对应的该词语出现的次数。

接下来,我们想要按照词的出现次数排序。第4章介绍字典的时候曾经提到,字典是没有办法排序的,我们需要把字典转换为列表,然后利用列表的sort()方法来排序。因为我们是使用人物的出现次数来排序的,所以要给sort()方法传递一个key参数,以指定用来进行比较的元素,该元素就是取自于可迭代对象中。这里调用了自定义的takeSecond()函数。这个函数接收的参数是一个列表,返回的是这个列表的第2个元素。这样,我们就可以指定第2个元素进行逆序排序,并且把结果赋值给items。

然后,借助range()函数生成一个等差数组,展示items中前20个元素。在每次循环中,我们先把获取的元素赋值给变量item。然后把item的第1个列表元素赋值给变量keyWord,第2个元素赋值给变量count。然后使用print()方法把格式化后的两个变量输出到屏幕上。这里我们用到了format()方法,它可以按照需求来格式化字符串。代码的含义是把keyWord的值左对齐,宽度是10;把count的值右对齐,宽度是5。

提示 字符串的format()方法可以用来格式化字符串。format()方法通过字符串中的花括号 {} 来识别要替换的内容,而format()中的参数是要填入的内容,按照顺序进行匹配。花括号中冒号后面的<符号表示左对齐,>符号表示右对齐,数字表示宽度。

然后调用main()函数。最终得到的词频统计结果如图17-6所示。

图17-6

我们发现一个问题,大部分的词语都是一个字。也就是没有把长度为1的词语进行筛选。没关系,在下一节中,我们继续优化这个程序。

17.5 筛选长度为1的词语

因为我们没有对分词的结果进行筛选,所以前20个高频词语大多是一个字,而这显然不是我们想要的结果。因为一个字的词语没有太多的含义,而我们需要的是有意义的词语,所以接下来,我们介绍如何将长度为1的字的词语过滤掉。先来看代码,为了便于区分,我们将新增的代码突出显示出来,代码参见ch17\17.3.py。

import jiebadef takeSecond(elem):return elem[1]def main():path = "西游记.txt"file = open(path,'r',encoding="utf-8")text=file.read()file.close()words = jieba.lcut(text)counts = {}for word in words:if len(word) == 1:continueelse:counts[word] = counts.get(word,0) + 1items = list(counts.items())items.sort(key = takeSecond,reverse=True)    for i in range(20):item=items[i]keyWord =item[0]count=item[1]print("{0:<10}{1:>5}".format(keyWord,count))main()

我们介绍一下突出显示的新增代码的含义。当通过for循环来遍历列表words中的每个元素时,在循环体中,增加了一个判断条件。如果表示每个词语的变量word的长度等于1,则直接进入下一次循环;否则,才会统计这个词语出现的次数。

运行程序,得到的词频统计结果如图17-7所示。

图17-7

现在得到的词语已经都是词组了,这个结果要比单个字的词语更有实际意义。但是,显然这样还是不够的,因为诸如“一个”“那里”和“怎么”这样的词组,对于我们理解西游记也没有什么帮助。在下一节中,我们会介绍如何去除这类不需要的词语。

17.6 去除不需要的词语

从上一节的词频统计结果可知,输出的结果中存在着一系列不需要的词语。其实,人名的词语对于分析小说剧情和主人公是最有帮助的,所以我们还要进一步优化程序,将不需要的词语进行过滤。具体代码如下所示,新增代码还是突出显示,代码参见ch17\17.4.py。

import jiebadef takeSecond(elem):return elem[1]def main():
path = "西游记.txt"
file = open(path,"r",encoding="utf-8")
text=file.read()
file.close()words = jieba.lcut(text)
counts = {}
for word in words:if len(word) == 1:continue    else:counts[word] = counts.get(word,0) + 1file = open("excludes.txt","r")excludes =file.read().split(",")file.closefor delWord in excludes:try:del counts[delWord]except:continueitems = list(counts.items())items.sort(key = takeSecond,reverse=True)    for i in range(20):item=items[i]keyWord =item[0]count=item[1]print("{0:<10}{1:>5}".format(keyWord,count))main()

这里介绍一下突出显示的代码的含义。我们把想要从高频词语结果中去除的词组放到一个叫作excludes.txt的文本文件中。可以从上一节的运行结果中进行筛选,从而得到这些词。

读取这个文件,并且对读取内容调用split方法,以“,”(逗号)作为分隔符,将结果赋值给一个叫作excludes的列表。现在excludes列表中的元素就是我们在上一节中想要从高频词语结果中去除的词组。

接下来,通过一个新的for循环遍历列表excludes中的元素,把每个元素赋值给变量delWord。在循环体中,使用del语句,将字典counts中的键为delWord的键—值对删除。

这里需要注意一下,如果字典中不包含所要删除的键,程序会报错。所以我们用到了异常处理的语句,当出现异常后,在except语句中使用continue语句跳转到下一次循环。

运行程序,得到的词频统计结果如图17-8所示。

图17-8

现在得到的结果已经把不需要的词组都去掉了。但是,我们发现结果还是有些瑕疵,因为“行者”“大圣”“老孙”和“悟空”都是指的“孙悟空”,而分开统计显然是不妥的。在下一小节,我们会介绍如何将同一个人名进行合并。

17.7 合并人名

从前面的词频统计结果可知,输出的结果中存在同一个人的多个称呼的词语。所以我们会进一步优化程序,将同一个人名的不同称呼进行合并。具体代码如下所示,新增代码突出显示出来了。代码参见ch17\17.5.py。

import jiebadef takeSecond(elem):return elem[1]def main():path = "西游记.txt"file = open(path,"r",encoding="utf-8")text=file.read()file.close()words = jieba.lcut(text)counts = {}for word in words:if len(word) == 1:continue    elif word == "大圣" or word=="老孙" or word=="行者" or word=="孙大圣" or word=="孙行者" or word=="猴王" or word=="悟空" or word=="齐天大圣" or word=="猴子":rword = "孙悟空"elif word == "师父" or word == "三藏" or word=="圣僧":rword = "唐僧"elif word == "呆子" or word=="八戒" or word=="老猪":rword = "猪八戒"elif word=="沙和尚":rword="沙僧"elif word == "妖精" or word=="妖魔" or word=="妖道":rword = "妖怪"elif word=="佛祖":rword="如来"elif word=="三太子":rword="白马"else:rword = wordcounts[rword] = counts.get(rword,0) + 1file = open("excludes.txt","r")excludes =file.read().split(",")file.closefor delWord in excludes:try:del counts[delWord]except:continueitems = list(counts.items())items.sort(key = takeSecond,reverse=True)    for i in range(20):item=items[i]keyWord =item[0]count=item[1]print("{0:<10}{1:>5}".format(keyWord,count))main()

来看一下新增代码含义。这里新建了一个变量rword,用它代替word作为字典counts的键。我们会将同一个人但是有多个称谓的词语变量word,重新赋值给变量rword。然后,将rword作为字典counts的键,将其出现次数作为值。例如,我们知道“师傅”“三藏”和“圣僧”都是对“唐僧”的尊称,所以我们可以这些词归类为唐僧。其他几个人物也都有重复的称谓,处理方法都是类似的。

运行程序,得到的词频统计结果如图17-9所示。

图17-9

现在,我们就可以得到《西游记》里出现最多的词语,其中“孙悟空”出现了6639次,是当之无愧的主角,紧随其后的是“唐僧”和“猪八戒”。作为各类反派的统称“妖怪”排名第四。“沙僧”位列第五,属于师徒四人中存在感最低的。这个统计结果和我们平常对《西游记》的认知是完全符合的。

可是,我们现在又有了一个新的问题,那就是现在得到的这个结果太不直观了,有没有更好的展示形式呢?在下一节中,我们会介绍如何用图表的方式来更加直观地展示结果。

17.8 用词云库(wordcloud)表示

在上一节中,我们已经得到了《西游记》中的高频词语。接下来,我们介绍如何用词云来展现这些信息。首先,我们要明白什么是词云。词云又叫作文字云,是对文本数据中出现频率较高的“关键词”在视觉上的突出呈现,把关键词的渲染成类似云一样的彩色图片,从而一眼就可以领略文本数据所要表达的主要意思。我们在本节借助的是wordcloud这个词云模块。

首先,我们需要安装wordcloud模块。还是使用pip工具,在命令行中输入“pip install wordcloud”,回车后开始安装,如图17-10所示。

图17-10

安装成功后,我们可以看到当前安装的wordcloud模块的版本是1.5.0,如图17-11所示。

图17-11

因为wordcloud模块要用到matplotlib模块,如果没有安装过这个模块,调用wordcloud就会报错,如图17-12所示。

图17-12

所以为了使用wordcloud,我们还需要安装matplotlib,安装方式如图17-13所示。

图17-13

在使用wordcloud时,要导入wordcloud模块,然后就可以使用wordcloud.WordCloud()方法来创建一个词云图。我们可以指定词云图的字体、图片大小以及背景色。

我们在西游记的分词代码中增加了生成词云图的功能,新增代码突出显示出来。代码参见ch17\17.6.py。

import jieba
import wordclouddef takeSecond(elem):return elem[1]def createWordCloud(text):w=wordcloud.WordCloud   (font_path="msyh.ttf",width=1000,height=500,background_color="white")w.generate(text)w.to_file("西游记词云图.jpg")def main():path = "西游记.txt"file = open(path,"r",encoding="utf-8")text=file.read()file.close()words = jieba.lcut(text)counts = {}for word in words:if len(word) == 1:continue    elif word == "大圣" or word=="老孙" or word=="行者" or word=="孙大圣" or word=="孙行者" or word=="猴王" or word=="悟空" or word=="齐天大圣" or word=="猴子":rword = "孙悟空"elif word == "师父" or word == "三藏" or word=="圣僧":rword = "唐僧"elif word == "呆子" or word=="八戒" or word=="老猪":rword = "猪八戒"elif word=="沙和尚":rword="沙僧"elif word == "妖精" or word=="妖魔" or word=="妖道":rword = "妖怪"elif word=="佛祖":rword="如来"elif word=="三太子":rword="白马"else:rword = wordcounts[rword] = counts.get(rword,0) + 1file = open("excludes.txt","r")excludes =file.read().split(",")file.closefor delWord in excludes:try:del counts[delWord]except:continueitems = list(counts.items())items.sort(key = takeSecond,reverse=True)    for i in range(20):item=items[i]keyWord =item[0]count=item[1]print("{0:<10}{1:>5}".format(keyWord,count))createWordCloud(str(items[0:20]))main()

正如前面所说的,首先需要导入wordcloud模块。然后我们新添加了一个名为createWordCloud()的自定义函数,它的参数就是要生成词云图的词组,我们把这个参数叫作text。在该函数中,首先生成了一个词云对象,指定字体的路径是“msyh.ttf”,图片的宽度是1000像素,高度是500像素,背景色是白色。然后调用generate方法生成词云,传入的参数就是要生成词云图的词组(即text)。然后,指定词云图输出到的文件。

提示 记住,如果词语中有中文,我们一定要设置字体路径,否则出来都是一个一个的方框,而不是文字,如图17-14所示。

图17-14

在main()函数中,调用了这个新增加的函数,并且将列表items中的前20个高频词转换为字符串,作为参数传递给这个函数。

运行程序,就可以在指定路径下得到我们的词云图,可以看到,越是高频词,字体就会更大更醒目,如图17-15所示。

图17-15

通过这个词云图,我们可以对于《西游记》中的重要人物一目了然,整个小说就是围绕着唐僧师徒四人展开的,他们的核心任务就是“取经”,他们会在取经路上遇到各种“妖怪”以及“小妖”,还好他们有重要的后援“菩萨”和“如来”。

到这里,我们对小说《西游记》的分词就结束了。你也可以去找一些自己喜欢的文章,使用jieba分词来分析文章中的主要人物和剧情。

本文摘自由人民邮电出版社异步社区出版的《Python少儿趣味编程》

  • 少儿编程畅销书作者李强老师教孩子学Python编程
  • 少儿编程入门教程图书,零基础自学Python
  • 全彩印刷,提供代码素材下载,方便亲子互动和少儿自学

Python简单易学,功能强大,是少儿学习编程的shou选语言。本书是少儿学习Python编程的趣味指南,全书共17章,按照由简到难、逐步深入的方式组织各章内容。本书从认识Python开始,首先介绍了Python的安装和IDLE的使用,然后依次介绍了变量、数字和字符串、列表、元组和字典、布尔类型等数据类型,以及条件、循环、异常和注释、函数、面向对象编程、文件操作等基础知识,并且通过实际案例讲解了海龟绘图、Pygame基础和游戏编程,以及Python在自然语言处理方面的应用。

Python库中,如何使用jieba模块来实现古典名著《西游记》的分词相关推荐

  1. 使用4G模块实现IOT_Python库中,如何使用jieba模块来实现古典名著《西游记》的分词...

    接下来,我们将学习如何使用jieba模块来实现古典名著<西游记>的分词,并且会将书中重点人物出场次数以图形化的方式显示出来,并进一步创建一个词云图. 17.4.1 读取文件 因为小说< ...

  2. matlab 写excel 慢_我在12w+的Python库中,发现了让Excel快到起飞的秘密......

    Amber | 作者 图片源自网络 在这篇文章里,小编向大家介绍了Excel在数据分析中的妙用.不知大家在看完后,有没有亲自动手去体验下呢?有没有遇到什么问题呢? 虽说Excel在处理小批量数据时的优 ...

  3. python pipline_python中sklearn的pipeline模块实例详解

    最近在看<深度学习:基于Keras的Python实践(魏贞原)>这本书,书中8.3创建了一个Scikit-Learn的Pipeline,首先标准化数据集,然后创建和评估基线神经网络模型,代 ...

  4. 利用python库中的turtle模块绘制自己的名字

    # 利用turtle模块绘制自己名字 --杨文博 import turtle as p p.pensize(5) p.pencolor("cornflowerblue") p.se ...

  5. python参考文献_[zotero/python]库中参考文献条目删除后,清除残留PDF的脚本

    更新:使用 @滏阳河边捉蚯蚓 https://zhuanlan.zhihu.com/p/41297136上获取系统PDF文件和zotero.sqlite文件的代码,在此感谢! 在zotero的libr ...

  6. python参考文献目录_[zotero/python]库中参考文献条目删除后,清除残留PDF的脚本

    # -*- coding: utf-8 -*- """Spyder Editor""" from __future__ import pri ...

  7. python库中的ssl.py

    python要支持https传输,就要用到ssl.py文件.加密协议也从SSL版本(SSLv2.SSLv3)升级为TLS版本(TLSv1.TLSv1_1.TLSv1_2.TLSv1_3)也就是说,当前 ...

  8. Python标准库中的pickle模块

     pickle  -  Python对象序列化. pickle模块实现了用于序列化和反序列化Python对象结构的二进制协议."pickle"是将Python对象层次结构转换为 ...

  9. 「Python 编程」编码实现网络请求库中的 URL 解析器

    相信各位 Python 开发者都用过 Requests 库,有些朋友还用过 WebSockets 库.这里回顾一下它们的基本用法,例如使用 Requests 库向目标网站发出 GET 请求: impo ...

最新文章

  1. Linux性能研究(总)
  2. GhostNet : 轻量级网络模型,性能超越MobileNetV3(CVRP2020, 华为诺亚)
  3. halcon区域腐蚀膨胀算子_超越halcon速度的二值图像的腐蚀和膨胀,实现目前最快的半径相关类算法(附核心源码)。...
  4. Memory Banks
  5. 【ARM】ARM汇编程序设计(一)
  6. 为什么和平精英无响应_什么和为什么
  7. 网易架构师深入讲解Java开发!BAT等大厂必问技术面试题
  8. mysql允许两个用户远程连接,配置MySQL服务允许用户远程连接
  9. Spring Boot提交表单信息
  10. 大屏监控系统实战(12)-10分钟投票增量曲线制作(一)
  11. Vue插槽(solt)简单案例
  12. JS数组的相关操作(循环、查找、过滤、排序等)
  13. linux关于tcp协议ack的实现--总结和公平性问题
  14. CAD的dwg格式转换为PDF
  15. pandas读取xlsx文件
  16. python实现根据前序序列和中序序列求二叉树的后序序列
  17. 比较不错的MaciOS软件论坛
  18. 「游戏建模」3DMAX节点材质编辑器详解
  19. 2021-2027全球与中国气溶胶粒径谱仪市场现状及未来发展趋势
  20. granite crushers in german

热门文章

  1. 社会责任·价值共创,中关村网络安全与信息化产业联盟对话网信企业家海泰方圆董事长姜海舟
  2. 服务器上的系统盘和数据盘,云服务器系统盘和数据盘区别
  3. 全新的Lync 2013为我们带来了什么?
  4. instandceof
  5. SqlServer数据库 时间类型字段添加默认值 设置时间格式
  6. 随机生成车牌号【python实现】
  7. 此前小编为大家介绍了女人吃鸡蛋的好处,想必大家对鸡蛋这种蛋类有了更多的了解。今天小编为大家介绍另一种蛋类——鸭蛋。鸭蛋又名鸭卵,是人们经常食用的一种蛋类食品,与鸡蛋营养相当,吃它的好处众多。那么女人吃
  8. 阿里巴巴国际站询盘转化率
  9. 【Get深一度】Microsoft Office 2013 Word中图片插入后变模糊解决办法
  10. emoji表情符号编码大全