如何用Python来理一理红楼梦里的那些关系


Posted in Python onAugust 14, 2019

前言

今天,一起用 Python 来理一理红楼梦里的那些关系

不要问我为啥是红楼梦,而不是水浒三国或西游,因为我也鉴定的认为,红楼才是无可争议的中国古典小说只巅峰,且不接受反驳!而红楼梦也是我多次反复品读的为数不多的小说,对它的感情也是最深的。

好了,不酸了,开干。

数据准备

红楼梦 TXT 文件一份

金陵十二钗 + 贾宝玉 人物名称列表

人物列表内容如下:

宝玉 nr
黛玉 nr
宝钗 nr
湘云 nr
凤姐 nr
李纨 nr
元春 nr
迎春 nr
探春 nr
惜春 nr
妙玉 nr
巧姐 nr
秦氏 nr

这份列表,同时也是为了做分词时使用,后面的 nr 就是人名的意思。

数据处理

读取数据并加载词典

with open("红楼梦.txt", encoding='gb18030') as f:
    honglou = f.readlines()
  jieba.load_userdict("renwu_forcut")
  renwu_data = pd.read_csv("renwu_forcut", header=-1)
  mylist = [k[0].split(" ")[0] for k in renwu_data.values.tolist()]

这样,我们就把红楼梦读取到了 honglou 这个变量当中,同时也通过 load_userdict 将我们自定义的词典加载到了 jieba 库中。

对文本进行分词处理并提取

tmpNames = []
  names = {}
  relationships = {}
  for h in honglou:
    h.replace("贾妃", "元春")
    h.replace("李宫裁", "李纨")
    poss = pseg.cut(h)
    tmpNames.append([])
    for w in poss:
      if w.flag != 'nr' or len(w.word) != 2 or w.word not in mylist:
        continue
      tmpNames[-1].append(w.word)
      if names.get(w.word) is None:
        names[w.word] = 0
      relationships[w.word] = {}
      names[w.word] += 1
  • 首先,因为文中"贾妃", "元春","李宫裁", "李纨" 混用严重,所以这里直接做替换处理。
  • 然后使用 jieba 库提供的 pseg 工具来做分词处理,会返回每个分词的词性。
  • 之后做判断,只有符合要求且在我们提供的字典列表里的分词,才会保留。
  • 一个人每出现一次,就会增加一,方便后面画关系图时,人物 node 大小的确定。
  • 对于存在于我们自定义词典的人名,保存到一个临时变量当中 tmpNames。

处理人物关系

for name in tmpNames:
    for name1 in name:
      for name2 in name:
        if name1 == name2:
          continue
        if relationships[name1].get(name2) is None:
          relationships[name1][name2] = 1
        else:
          relationships[name1][name2] += 1

对于出现在同一个段落中的人物,我们认为他们是关系紧密的,每同时出现一次,关系增加1.

保存到文件

with open("relationship.csv", "w", encoding='utf-8') as f:
    f.write("Source,Target,Weight\n")
    for name, edges in relationships.items():
      for v, w in edges.items():
        f.write(name + "," + v + "," + str(w) + "\n")

  with open("NameNode.csv", "w", encoding='utf-8') as f:
    f.write("ID,Label,Weight\n")
    for name, times in names.items():
      f.write(name + "," + name + "," + str(times) + "\n")
  • 文件1:人物关系表,包含首先出现的人物、之后出现的人物和一同出现次数
  • 文件2:人物比重表,包含该人物总体出现次数,出现次数越多,认为所占比重越大。

制作关系图表

使用 pyecharts 作图

def deal_graph():
  relationship_data = pd.read_csv('relationship.csv')
  namenode_data = pd.read_csv('NameNode.csv')
  relationship_data_list = relationship_data.values.tolist()
  namenode_data_list = namenode_data.values.tolist()

  nodes = []
  for node in namenode_data_list:
    if node[0] == "宝玉":
      node[2] = node[2]/3
    nodes.append({"name": node[0], "symbolSize": node[2]/30})
  links = []
  for link in relationship_data_list:
    links.append({"source": link[0], "target": link[1], "value": link[2]})

  g = (
    Graph()
    .add("", nodes, links, repulsion=8000)
    .set_global_opts(title_opts=opts.TitleOpts(title="红楼人物关系"))
  )
  return g

首先把两个文件读取成列表形式

对于“宝玉”,由于其占比过大,如果统一进行缩放,会导致其他人物的 node 过小,展示不美观,所以这里先做了一次缩放

最后得出的关系图

如何用Python来理一理红楼梦里的那些关系

所有代码已经上传至 Github

最后,我还准备了一份更加全面的红楼人物字典,可以在代码仓库中找到-“renwu_total”,感兴趣的小伙伴也可以尝试下,制作一个全人物的关系图。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
一个基于flask的web应用诞生 用户注册功能开发(5)
Apr 11 Python
Python2.7编程中SQLite3基本操作方法示例
Aug 09 Python
LRUCache的实现原理及利用python实现的方法
Nov 21 Python
python机器人行走步数问题的解决
Jan 29 Python
pygame实现俄罗斯方块游戏
Jun 26 Python
Python中BeautifuSoup库的用法使用详解
Nov 15 Python
Python2和Python3中@abstractmethod使用方法
Feb 04 Python
python实现电子词典
Mar 03 Python
Python requests模块cookie实例解析
Apr 14 Python
Python 多线程共享变量的实现示例
Apr 17 Python
Python实现电视里的5毛特效实例代码详解
May 15 Python
如何在Python3中使用telnetlib模块连接网络设备
Sep 21 Python
django之自定义软删除Model的方法
Aug 14 #Python
python实现登录密码重置简易操作代码
Aug 14 #Python
python 定时器每天就执行一次的实现代码
Aug 14 #Python
Django 项目重命名的实现步骤解析
Aug 14 #Python
Django如何实现网站注册用户邮箱验证功能
Aug 14 #Python
python 叠加等边三角形的绘制的实现
Aug 14 #Python
Django为窗体加上防机器人的验证码功能过程解析
Aug 14 #Python
You might like
php中的动态调用实例分析
2015/01/07 PHP
php+mysql实现简单的增删改查功能
2015/07/13 PHP
thinkPHP下ueditor的使用方法详解
2015/12/26 PHP
redis+php实现微博(三)微博列表功能详解
2019/09/23 PHP
浅谈javascript中的作用域
2012/04/07 Javascript
jquery插件开发之实现md5插件
2014/03/17 Javascript
JavaScript实现跑马灯抽奖活动实例代码解析与优化(二)
2016/02/16 Javascript
jQuery插件编写步骤详解
2016/06/03 Javascript
Javascript中this关键字指向问题的测试与详解
2017/08/11 Javascript
简单实现js进度条加载效果
2020/03/25 Javascript
Three.js中网格对象MESH的属性与方法详解
2017/09/27 Javascript
微信小程序异步API为Promise简化异步编程的操作方法
2018/08/14 Javascript
Vue实现table上下移动功能示例
2019/02/21 Javascript
微信小程序实现侧边栏分类
2019/10/21 Javascript
JS实现简单省市二级联动
2019/11/27 Javascript
pycharm 使用心得(八)如何调用另一文件中的函数
2014/06/06 Python
pandas.DataFrame选取/排除特定行的方法
2018/07/03 Python
python中的tcp示例详解
2018/12/09 Python
python实现京东订单推送到测试环境,提供便利操作示例
2019/08/09 Python
Python函数的迭代器与生成器的示例代码
2020/06/18 Python
python 自定义异常和主动抛出异常(raise)的操作
2020/12/11 Python
python爬取youtube视频的示例代码
2021/03/03 Python
CSS3为背景图设置遮罩并解决遮罩样式继承问题
2020/06/22 HTML / CSS
俄罗斯隐形眼镜和眼镜在线商店:Cronos
2020/06/02 全球购物
几个Shell Script面试题
2012/08/31 面试题
实习自我鉴定模板
2013/09/28 职场文书
遗体告别仪式答谢词
2014/01/23 职场文书
教师远程培训感言
2014/03/06 职场文书
试用期员工工作自我评价
2014/09/10 职场文书
入党积极分子学习优秀共产党员先进事迹思想汇报
2014/09/13 职场文书
2014乡镇党政班子四风问题思想汇报
2014/09/14 职场文书
干部作风建设个人剖析材料
2014/10/11 职场文书
干部四风问题整改措施思想汇报
2014/10/13 职场文书
2015圣诞节贺卡寄语
2015/03/24 职场文书
小学语文教学随笔
2015/08/14 职场文书
Redis 哨兵集群的实现
2021/06/18 Redis