【关于 KBQA】那些你不知道的事

共 9876字,需浏览 20分钟

 ·

2021-03-24 13:16

作者:杨夕

项目地址:https://github.com/km1994/nlp_paper_study

个人介绍:大佬们好,我叫杨夕,该项目主要是本人在研读顶会论文和复现经典论文过程中,所见、所思、所想、所闻,可能存在一些理解错误,希望大佬们多多指正。


一、基于词典和规则的方法

1.1 介绍

1.1.1 开源知识图谱

工业界的知识图谱有两种分类方式,第一种是根据领域的覆盖范围不同分为通用知识图谱和领域知识图谱。其中通用知识图谱注重知识广度,领域知识图谱注重知识深度。通用知识图谱常常覆盖生活中的各个领域,从衣食住行到专业知识都会涉及,但是在每个领域内部的知识体系构建不是很完善;而领域知识图谱则是专注于某个领域(金融、司法等),结合领域需求与规范构建合适的知识结构以便进行领域内精细化的知识存储和问答。代表的知识图谱分别有:

  • 通用知识图谱

    • Google Knowledge Graph

    • Microsoft Satori & Probase

  • 领域知识图谱

    • Facebook 社交知识图谱

    • Amazon 商品知识图谱

    • 阿里巴巴商品知识图谱

    • 上海交大学术知识图谱

第二种分类方式是按照回答问题需要的知识类别来定义的,分为常识知识图谱和百科全书知识图谱。针对常识性知识图谱,我们只会挖掘问题中的词之间的语义关系,一般而言比较关注的关系包括 isA Relation、isPropertyOf Relation,问题的答案可能根据情景不同而有不同,所以回答正确与否往往存在概率问题。而针对百科全书知识图谱,我们往往会定义很多谓词,例如DayOfbirth, LocatedIn, SpouseOf 等等。这些问题即使有多个答案,这些答案往往也都是确定的,所以构建这种图谱在做问答时最优先考虑的就是准确率。代表的知识图谱分别有:

  • 常识知识图谱

    • WordNet, KnowItAll, NELL, Microsoft Concept Graph

  • 百科全书知识图谱

    • Freebase, Yago, Google Knowledge Graph

1.1.2 代表项目

  1. 豆瓣影评问答

  2. 基于医疗知识图谱的问答系统

1.2 流程

1.2.1. 句子输入

- eg:query:高血压要怎么治?需要多少天?

1.2.2. 问句解析

  1. 实体抽取:

  • 作用:得到匹配的词和类型

  • 方法:

  • 模式匹配:

    • 介绍:主要采用规则 提槽

    • 工具:正则表达式

  • 词典:

    • 介绍:利用词典进行匹配

    • 采用的词典匹配方法:trie和Aho-Corasick自动机,简称AC自动机

    • 工具:ahocorasick、FlashText 等 python包

  • 基于词向量的文本相似度计算:

    • 介绍:计算 query 中 实体 与 实体库 中候选实体的相似度,通过设定阈值,得到最相似的 实体

    • 工具:词向量工具(TF-idf、word2vec、Bert 等)、相似度计算方法(余弦相似度、L1、L2等)

  • 命名实体识别方法:

    • 利用 命名实体识别方法 识别 query 中实体

    • 方法:BiLSTM-CRF等命名实体识别模型

  • 举例说明:

eg:通过解析 上面的 query ,获取里面的实体和实体类型:{'Disease': ['高血压'], 'Symptom': ['高血压'], 'Complication': ['高血压']}

  1. 属性和关系抽取:

  • 作用:抽取 query 中 的 属性和关系

  • 方法:

  • 模式匹配:

    • 介绍:主要采用规则匹配

    • 工具:正则表达式

  • 词典:

    • 介绍:利用词典进行匹配

    • 采用的词典匹配方法:trie和Aho-Corasick自动机,简称AC自动机

    • 工具:ahocorasick、FlashText 等 python包

  • 意图识别方法:

    • 介绍:采用分类模型 对 query 所含关系 做预测

    • 工具:

      • 机器学习方法:LR、SVM、NB

      • 深度学习方法:TextCNN、TextRNN、Bert 等

  • 命名实体识别方法:【同样,可以采用命名实体识别挖掘出 query 中的某些动词和所属类型】

    • 利用 命名实体识别方法 识别 query 中实体

    • 方法:BiLSTM-CRF等命名实体识别模型

  • 举例说明:

- eg:通过解析 上面的 query ,获取里面的实体和实体类型:
- predicted intentions:['query_period'] 高血压要怎么治?
- word intentions:['query_cureway'] 需要多少天?

1.2.3. 查询语句生成

  • 作用:根据 【问句解析】 的结果,将 实体、属性和关系转化为对于的 图数据库(eg:Neo4j图数据库等)查询语句

  • 举例说明:

- eg:
对于 query:高血压要怎么治?需要多少天?
- sql 解析结果:
[{'intention': 'query_period', 'sql': ["MATCH (d:Disease) WHERE d.name='高血压' return d.name,d.period"]}, {'intention': 'query_cureway', 'sql': ["MATCH (d:Disease)-[:HAS_DRUG]->(n) WHERE d.name='高血压' return d.name,d.treatment,n.name"]}]

1.2.4. 查询数据库和结果生成

  • 作用:利用 【查询语句生成】 的结果,去 图数据库 中 查询 答案,并利用预设模板 生成答案

- eg:
- 高血压可以尝试如下治疗:药物治疗;手术治疗;支持性治疗

二、基于信息抽取的方法

2.1 介绍

2.1.1 开源知识图谱介绍


2.2 流程

2.2.1. 分类单跳和多跳问句

  • 思路:利用 文本分类方法 对 query 进行分类,判断其属于 一跳问题还是多跳问题

  • 方法:文本分类方法【TextCNN、TextRNN、Bert 等】

  • 解析

单跳:SPARQL 只出现一个三元组

q26:豌豆公主这个形象出自于哪?
select ?x where { <豌豆公主_(安徒生童话)> <作品出处> ?x. }
<安徒生童话>

双跳或多跳:SPARQL 只出现两个以上三元组

q524:博尔赫斯的国家首都在哪里?
select ?x where { <豪尔赫·路易斯·博尔赫斯_(阿根廷作家)> <出生地> ?y. ?y <首都> ?x}
<布宜诺斯艾利斯_(阿根廷的首都和最大城市)>

2.2.2. 分类链式问句(二分类)

  • 思路:利用 文本分类方法 对 query 进行分类,判断其是否 属于 链式问句

  • 介绍:链式:SPARQL 多个三元组呈递进关系,x->y->z,非交集关系

q894:纳兰性德的父亲担任过什么官职?
select ?y where { <纳兰性德> <父亲> ?x. ?x <主要职位> ?y. }
"武英殿大学士" "太子太傅"
q554:宗馥莉任董事长的公司的公司口号是?
select ?y where { ?x <董事长> <宗馥莉>. ?x <公司口号> ?y. }
"win happy health,娃哈哈就在你身边"

2.2.3. 主谓宾分类(三分类)

  • 思路:利用 文本分类方法 对 query 进行分类,判断 问句的答案对应三元组里面的 主谓宾

  • 问句的答案对应三元组里面的主语,spo=0

q70:《悼李夫人赋》是谁的作品?
select ?x where { ?x <代表作品> <悼李夫人赋>. }
<汉武帝_(汉朝皇帝)>

  • 问句的答案对应三元组里面的谓语,spo=1

q506:林徽因和梁思成是什么关系?
select ?x where { <林徽因_(中国建筑师、诗人、作家)> ?x <梁思成>. }
<丈夫>

  • 问句的答案对应三元组里面的宾语,spo=2

q458:天津大学的现任校长是谁?
select ?x where { <天津大学> <现任校长> ?x . }
<李家俊_(天津市委委员,天津大学校长)>

2.2.4. 实体提及(mention)识别

  • 思路:对于 给定 query,我们需要 识别出 query 中 所含有的 实体提及(mention)

  • 问题:主办方提供的数据中 包含 :query、sql语句,但是 sql语句中的实体并不能在 query 中被找到,如下:

q1440:济南是哪个省的省会城市?
select ?x where { ?x <政府驻地> <济南_(山东省省会)>. }
<山东_(中国山东省)>

注:query 中 的 济南 是 <济南_(山东省省会)>的 简称、省会城市 在 知识库中 对应 <政府驻地>

  • 解决方法:根据训练语料的SPARQL语句,查找实体的提及,反向构建训练数据

q6:叔本华信仰什么宗教?
select ?y where { <亚瑟·叔本华> <信仰> ?y. }
<佛教>

>>>
叔本华信仰什么宗教? ['叔本华']
亚瑟·叔本华信仰什么宗教? ['亚瑟·叔本华']

>>>
叔 本 华 信 仰 什 么 宗 教 ? B-SEG I-SEG E-SEG O O O O O O O
亚 瑟 · 叔 本 华 信 仰 什 么 宗 教 ? B-SEG I-SEG I-SEG I-SEG I-SEG E-SEG O O O O O O O

2.2.5. 关系分类 (语义相似度计算,二分类问题)

  • 目标:查询实体关系中与问句最相似的关系

  • 思路:

    • 正例:根据给定 训练集,获得 实体和关系 的样本

    • 负例:根据 该实体名 从 Neo4j 图数据库中随机抽取出 5个关系作为负例

q1:莫妮卡·贝鲁奇的代表作?
select ?x where { <莫妮卡·贝鲁奇> <代表作品> ?x. }
<西西里的美丽传说>

>>> 表达形式一:
正例:
莫妮卡·贝鲁奇的代表作?代表作品 1
负例:
莫妮卡·贝鲁奇的代表作?出生地 0
莫妮卡·贝鲁奇的代表作?类型 0
莫妮卡·贝鲁奇的代表作?制片地区 0
莫妮卡·贝鲁奇的代表作?主演 0
莫妮卡·贝鲁奇的代表作?作者 0


2.2.6. 实体链指 【实体消歧】

  • 问题:对于 问句中的实体提及在 Neo4j 图数据库 中可能存在多个相关实体,如何选取 将 实体提及 链指到 对应的 知识库中的实体

q15:清明节起源于哪里?
select ?x where { <清明_(二十四节气之一)> <起源> ?x. }
<绵山风景名胜区>

>>>
问句中 实体提及:清明
对应的知识库中的 实体:<清明_(二十四节气之一)>、清明_(汉语词汇)、清明_(长篇小说)、清明_(唐代杜牧诗作)等

  • 目标:查找问句中实体提及对应的唯一实体

  • 思路:

    • 在训练集上,令标注的实体标签为1,其余候选实体标签为0,使用逻辑回归对上述特征进行拟合。

    • 在验证集和测试集上,使用训练好的机器模型对每个实体打分,保留分数排名前n的候选实体。

  • 特征选择:

# 获取 mention 的特征 mention_features : [mention, f1, f2, f3]
# f1 : mention的长度
# f2 : mention 在 SogouLabDic.dic 中 的词频
# f3 : mention 在 question 中 的位置
def get_mention_feature(self,question,mention):
f1 = float(len(mention)) #mention的长度
try:
f2 = float(self.word_2_frequency[mention]) # mention的tf/10000
except:
f2 = 1.0
if mention[-2:] == '大学':
f2 = 1.0
try:
f3 = float(question.index(mention))
except:
f3 = 3.0
#print ('这个mention无法提取位置')
return [mention,f1,f2,f3]

  1. 图谱子图与问题的匹配度:计算问题和主语实体及其两跳内关系间的相似度

    1. 实体提及及两跳内关系和实体与问题重叠词数量

    2. 实体提及及两跳内关系和实体与问题重叠字数量

# similar_features : [overlap,jaccard] * {(q_tokens,e_tokens), (q_chars,e_chars), (q_tokens,p_tokens), (q_chars,p_chars)} = 8

def extract_subject(self,entity_mentions,subject_props,question):
...
#得到实体两跳内的所有关系
entity = '<'+entity+'>'
if entity in self.entity2hop_dic:
relations = self.entity2hop_dic[entity]
else:
relations = kb.GetRelations_2hop(entity)
self.entity2hop_dic[entity] = relations
# 计算问题和主语实体及其两跳内关系间的相似度
similar_features = ComputeEntityFeatures(question,entity,relations)
...

def ComputeEntityFeatures(question,entity,relations):
'''
抽取每个实体或属性值2hop内的所有关系,来跟问题计算各种相似度特征
input:
question: python-str
entity: python-str <entityname>
relations: python-dic key:<rname>
output:
[word_overlap,char_overlap,word_embedding_similarity,char_overlap_ratio]
'''
#得到主语-谓词的tokens及chars
p_tokens = []
for p in relations:
p_tokens.extend(segger.cut(p[1:-1]))
p_tokens = [token[0] for token in p_tokens]
p_chars = [char for char in ''.join(p_tokens)]

q_tokens = segger.cut(question)
q_tokens = [token[0] for token in q_tokens]
q_chars = [char for char in question]

e_tokens = segger.cut(entity[1:-1])
e_tokens = [token[0] for token in e_tokens]
e_chars = [char for char in entity[1:-1]]

qe_feature = features_from_two_sequences(q_tokens,e_tokens) + features_from_two_sequences(q_chars,e_chars)
qr_feature = features_from_two_sequences(q_tokens,p_tokens) + features_from_two_sequences(q_chars,p_chars)
#实体名和问题的overlap除以实体名长度的比例
return qe_feature+qr_feature


def features_from_two_sequences(s1,s2):
#overlap
overlap = len(set(s1)&(set(s2)))
#集合距离
jaccard = len(set(s1)&(set(s2))) / len(set(s1)|(set(s2)))
#词向量相似度
#wordvecsim = model.similarity(''.join(s1),''.join(s2))
return [overlap,jaccard]

  1. 实体提及的流行度特征

    1. 实体提及在图谱中关系个数/出现频率

# popular_feature : GetRelationNum = 1 实体的流行度特征
def extract_subject(self,entity_mentions,subject_props,question):
...
#实体的流行度特征
popular_feature = kb.GetRelationNum(entity)
...


def GetRelationNum(self,entity):
'''根据实体名,得到与之相连的关系数量,代表实体在知识库中的流行度'''
cql= "match p=(a:Entity)-[r1:Relation]-() where a.name=$name return count(p)"
res = self.session.run(cql,name=entity)
ans = 0
for record in res:
ans = record.values()[0]
return ans

  1. 实体名称和问题的字符串匹配度(char/word)

  2. 问题和实体提及语义相似度

  3. 问题和实体关系的最大相似度 ...

  • 标签:是否为 对应实体

q15:清明节起源于哪里?
select ?x where { <清明_(二十四节气之一)> <起源> ?x. }
<绵山风景名胜区>

>>>
清明节起源于哪里?<清明_(二十四节气之一)> 1.0 1.0 0.43 0.15978466 0.6 0.99257445 ... 0
清明节起源于哪里?<清明_(汉语词汇)> 0.9 1.0 0.43 0.97132427 0.58 0.99660385 ... 1
清明节起源于哪里?<清明_(长篇小说)> 0.8 1.0 0.43 0.920861 0.38 0.0007164952 ... 0
清明节起源于哪里?<清明_(唐代杜牧诗作)> 0.8 1.0 0.43 0.920861 0.38 0.0007164952 ... 0
...

  • 分类方法:

    • LR、SVM、xgboost 等


q15:清明节起源于哪里?
select ?x where { <清明_(二十四节气之一)> <起源> ?x. }
<绵山风景名胜区>

>>> 实体提及 和 关系
实体提及 <清明_(二十四节气之一)>
关系 起源

>>> (entity,relation)
(entity,relation):(<清明_(二十四节气之一)>,起源 )

>>> 候选查询路径:(entity,relation) => entity 的 relation
(<清明_(二十四节气之一)>,起源 ): 清明_(二十四节气之一)的起源

>>> 文本匹配:
清明节起源于哪里?清明_(二十四节气之一)的起源 1
清明节起源于哪里?清明_(二十四节气之一)的类型 0
清明节起源于哪里?清明_(汉语词汇)的起源地 0
清明节起源于哪里?清明_(汉语词汇)的类型 0

2.2.8. 实体桥接及答案检索

  • 动机:

    • 2.2.7 节 所提方法只能处理 单实体问题,但是 样本中 不仅 包含 单实体问题,还包含 多实体问题,那需要怎么解决呢?

eg:北京大学出了哪些哲学家

  • 解决方法:实体桥接

  • 思路:

    • 对于每个问题,首先对2.2.7 节 打分后的候选查询路径进行排序,保留前30个单关系的查询路径(entity1,relation1)。

    • 对于这些查询路径,到知识库中进行检索,验证其是否能和其他候选实体组成多实体情况的查询路径(entity1,relation1,ANSWER,relation2,entity2),将其加入候选查询路径中。

    • 最后,本文将2.2.7 节 单实体情况排名前三的候选查询路径和本节得到的双实体情况查询路径同时和问题计算重叠的字数,选择重叠字数最多的作为最终的查询路径,认为其在语义和表达上最与问题相似。

参考资料

  1. 中文知识图谱问答 CCKS2019 CKBQA - 参赛总结

  2. ccks2019-ckbqa-4th-codes

  3. QA-Survey

  4. 中文知识图谱问答 CCKS2019 CKBQA 参赛总结

  5. CCKS2019 测评报告




浏览 80
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报