zoukankan      html  css  js  c++  java
  • 基于KB的QA系统学习记录

    此为大三上学期的nlp项目之问答系统,此篇博客用来记录我的学习过程

    阅读

    揭开知识库问答KB-QA的面纱1·简介篇

    阅读链接

    笔记:

    为什么需要知识图谱?

    因为智能的实现离不开知识,而人类的知识往往采用非结构化的自然语言的形式,不适合计算机处理,于是创建了知识图谱,即一种利于计算机理解和处理的知识库

    知识图谱是如何获取的呢?

    知识图谱的获取当然是通过蕴含知识的自然语言来获取。根据知识源的不同,知识图谱可以分为Curated KBs 和 Extracted KBs 两种。前者的知识源为维基百科这些比较好的数据源,抽取难度小,质量好,缺点是数量相对较少;后者的知识源为整个互联网,虽然数据量庞大,但是质量相对较差。

    从蕴含知识自然语言到知识图谱的转化涉及到实体链接(Entity Linking)和关系抽取(relation extraction)

    实体链接:实体链接即将自然语言中的实体与知识库中的实体对应起来的一个过程,举个例子,假设知识库中有周恩来这个结点,这时候在自然语言文本中的得到一个实体周总理,我们把他加入数据库中,由于周总理值得就是知识库中的周主席,所以我们不需要再添加新的结点。这对知识图谱的构建有很大的意义,能让其质量更好

    关系抽取:关系抽取就是要把自然语言中实体的关系抽取出来,比如"小明和小红是好朋友",实体小明和小红直接的关系为朋友,我们可以用三元组<小明,朋友,小红>来表示

    术语:

    问题:

    启发:

    TO DO.....................................

    项目

    基于refo+知识图谱的问答系统

    项目链接
    该项目的基本思路
    (1)用户问题转换为Word对象序列
    Word类

    class Word:
        def __init__(self, token, pos):
            self.token = token
            self.pos = pos
    

    该类有token和pos两个熟悉,用来表示一个word

    Tagger类

    class Tagger:
        def __init__(self, dict_paths=None):
            if dict_paths == None:
                return
            #加载自定义词典
            for p in dict_paths:
                jieba.load_userdict(p)
        
        #将句子(字符串)变成Word对象序列
        def getWordObjects(sentence):
            return [Word(token, pos) for token, pos in 
            jieba.posseg.cut(sentence)]
    

    该类可以通过jieba分词,对句子进行词性标注,使得字符串变成Word对象序列

    (2)将Word对象序列转换为知识图谱相应的查询语句

    通过对象正则表达式对Word对象进行匹配,匹配成功则能生产相应的查询语句

    (3)将查询语句应用于知识图谱中来返回答案

    改进的地方/需要的任务:

    (1)使用neo4j的知识图谱, neo4j的表达能力更强大

    (2)能不能让对象序列不是简单词性标准序列,使之成为语义序列。(不能,如果这样做了,那就不需要正则表达式了)

    注意:我后来才发现这个项目是浙江大学给入门级的u学生联手用的,意味他特别简单,同时方法也非常的奇葩(就是利用正则表达式对问句做语法上的匹配),与目前主流的方法不服(主流方法要上升到语义上),所以没法有是什么改进空间。

    基于jieba分词和贝叶斯分类的问答系统

    项目链接
    基本流程如图:

    (1)对问题进行实体抽取:这里他利用jieba实现,具体来说就是先定义好电影名、演员名、电影类型名的自定义词典(电影名nm,演员名nnt,电影类型名ng),

    然后通过jieba的词性标注的功能,通过词性就可以把电影名、演员名、电影类型名抽取出来。
    (2)对问题进行属性映射:这里他先构建好问题-问题模板训练集,


    通过贝叶斯分类器进行训练,使得可以把问题分类到相应的问题模板中

    (3)通过前面抽取出来的实体和问题类型,就可以产生相应的查询语句,进而通过知识图谱得到答案

    缺点不足:

    (1)实体抽取过于简单,缺乏灵活性,对于电影知识图谱来说,因为问题的实体无非就是电影、演员、类型三者之一,所以确实可以通过先构建实体词典再用jieba词性标注的方法找到问题的实体,但是这个方法却缺乏了一般性和灵活性,如果换一个知识图谱,不仅需要重新手动建立实体词典,同时效果或许不能那么好。
    (2)属性抽取使用的是贝叶斯分类的方法,需要我们手动建立问题——问题模板训练集,同样是缺乏了灵活性和一般性,如果要将该模型应用与其他的知识图片上,就需要重新手动建立问题——问题模板训练集,费时费力, 同时可回答的问题的范围是固定的,没法解决问题模板外的问题。

    基于bert的问答系统

    项目链接

    实践

    项目链接

    知识图谱的构建

    参考链接
    这里我们建立了一个简单的金融领域的知识图谱,建立的过程步骤如下:

    网页信息到可导入neo4j的csv文件

    我们选择了金融网站同花顺F10, 该网站记录了上市公司的各种信息,如图所示:

    通过网络爬虫我们得到了三个可导入neo4j的csv文件,分别为:
    概念结点文件concept.csv

    企业结点文件stock.csv

    高管结点文件executive.csv

    行业结点文件industry.csv

    高管—企业关系文件executive_stock.csv

    企业-概念关系文件stock_concept.csv

    企业-行业关系文件stock-industry.csv

    可导入neo4j的csv文件到neo4j知识图谱

    (1)首先将得到的csv文件统一放入neo4j的安装目录下的import目录

    (2)然后命令行进入neo4j安装目录下的bin目录,输入命令neo4j-admin import --mode=csv --database=mygraph2.db --nodes ../import/concept.csv --nodes ../import/executive.csv --nodes ../import/industry.csv --nodes ../import/stock.csv --relationships ../import/executive_stock.csv --relationships ../import/stock_concept.csv --relationships ../import/stock_industry.csv就可以将csv文件转换为知识图谱,得到的知识图谱如图所示:

    建立问答系统

    问答系统的整体结构图如下所示:

    解释:用户问题先进行实体抽取来得到抽取后的问题和实体类型:实体字典,抽取后的问题通过属性映射得到问题模板(编号),然后用实体类型:实体字典和问题模板(编号)来得到图形数据库相应的查询语言,最后将查询到结果进行语言的上的润色得到相应的答案。
    下面对每一个步骤做更具体的描述

    用户问题清洗

    用户的输入的问题总是会有很多的"杂质",比如一些不必要的标点符号等等,会影响后面的处理,所以我们需要进行清洗得到干净的问题。
    例如:用户输入的问题“华为,是什么?”清洗后得到“华为是什么”

    实体抽取

    实体抽取采用jieba自定义词典再分词的方法。
    (1)对于建立的金融数据库,其关键实体一般为高管、企业、概念、行业,于是将前面得到的可导入的csv文件中的所有的高管、企业、概念、行业抽取出来,分别给予ne、ns、nc、ni词性, 最后得到自定义词典mydict.txt,内容如下

    余劲 15 ne
    周学军 15 ne
    艾华集团 15 ns
    亿帆医药 15 ns
    IPV6概念 15 nc
    前海概念 15 nc
    印刷包装 15 ni
    家电行业 15 ni
    ...
    

    使用jieba.load_usrdict(mydict_txt)的方法将自定义词典导入jieba分词中
    (2)使用jieba分词工具的词性标注方法得到实体类型:实体字典和实体抽取后的问题
    若问题中有上面自定义词典中的实体,就会被词性标记为ne、ns、nc或ni,我们将该词性和对应的实体存入字典中,再将原先的问题中的抽出的关键实体用词性替换,得到抽取后的问题序列。下面举一个具体的例子。
    问题“华为是什么”会被jieba分词标注为[('华为', 'ns'), ('是', 'v'), ('什么', 'r')], 进而我们得到字典{'ns', 华为} 和 实体抽取后问题 “nr 是 什么”

    属性映射

    接下来需要将实体抽取后的问题映射到具体的问题模板(编号),我们采用分类的方法。
    (1)首先建立训练集, X_train是问题,y_train是对应的模板编号

    '''
    问题模板编号-问题模板
    0:企业的介绍
    1:高管的介绍
    ...
    '''
    X_train = ["ne 是 什么", "ne 的 介绍", "ne 的 个人信息", "ns 是 什么", "ns 怎么样"...]
    y_train = [1, 1, 1, 0, 0...]
    

    (2)接下来对X_train中的每一个数据进行文本特征的提取,我们使用了sklearn中的tfidvectorizer文本特征处理器,举例:
    “ne 是 什么” 通过文本特征提取会得到 [0.70710678 0.70710678]
    (3)将特征提取后的X_train, y_train输入分类器中进行训练得到问题-问题模板编号分类器, 这里我们使用贝叶斯分类器来进行分类

    查询语句cql

    前面我们得到了问题模板编号和实体类型:实体字典,下面就可以用它们来得到相应的查询语句cql。
    比如,前面我们得到了问题模板编号0和{'ns':华为字典}, 进而可以得到如下查询语句

    match (p:企业) where p.name='华为' return p.code  
    ...  
    

    语言润色

    最终将查询到结果进行整合、润色就可以得到答案


    最后用一张实例图来总结上面的流程

    效果展示

    不足和改进

    由于时间关系,暂时没有时间对这个系统进行改进,所以先将该系统的不足和改进放在这里
    (1)
    不足:实体抽取缺少灵活性,我们采用的jieba词性标注来抽取实体的方法,一个前提就是问题的关键实体必须在自定义的词典中并且要一模一样,然而很多不同名称的实体其实指向同一个实体,我们却将他们看做不同的实体
    改进:使用实体链接的方法将不同名称的实体映射到同一个实体,我想能不能进行词的相似度计算,在自定义词典中找到最相近的词作为实体。在网上了解到BERT+CRF的命名实体识别的效果比较好
    (2)
    不足:图形数据库的建立的比较简单
    改进:使用一些好的技术来建立更大更复杂的知识图谱
    (3)
    不足:该系统进行属性映射的方式,我们需要手动建立训练集来进行分类,当问题变得复杂,这项工作将变得费时费力,同时问题的回答范围由自己建立的问题模板的数量决定,具有一定的局限性。
    改进:从网络上了解到可以通过实体得到一个子图,然后再子图中寻找与问题最匹配的答案(进行相似度计算)。

  • 相关阅读:
    常用API及异常
    支付宝支付流程以及二次封装alipay包 支付模块后台接口的实现 订单接口实现
    课程页前台组件 课程模块的实现 表的设计及数据录入 课程详情页前台组件 课程详情页后台实现
    celery使用
    登录、注册、页头页面 手机号验证接口 发送短信接口 注册接口 多方式登录接口 手机号登录接口 发送短信接口频率限制 异常处理修订 接口缓存
    redis基础使用 python操作redis django-redis使用
    短信接口的使用(腾讯云)
    主页设计:页面头部组件 页面底部组件 轮播图组件 主页组件 主页后台: home模块创建 路由分发 Banner表的model设计 数据库迁移 注册模块
    前台vue环境配置 vue项目的创建 项目目录的重构 全局样式文件 配置文件 axios配置 cookies配置 element-ui配置 bootstrap和jquery的配置
    thinkphp6使用最新版本composer后多应用模式提示路由不存在
  • 原文地址:https://www.cnblogs.com/Serenaxy/p/14063869.html
Copyright © 2011-2022 走看看