FAQ(FAQ,frequently-asked questions)问答系统表示常见问题问答系统,常用于一些特定领域的智能客服,将用户经常问到的高频问答对索引起来,当新的提问命中时可以快速回答,准确而高效。
本文介绍一个简单的FAQ问答系统实现。基于检索和排序的两阶段框架,检索阶段基于Elasticsearch检索引擎、排序阶段基于语义匹配深度学习模型。后端基于SpringBoot系列框架。
FAQ问答大概的对话流程示意图如下:

系统的大致框架如下图所示:

以对话为例说一下系统各个模块的协同:
用户问题向后台发送HTTP请求;用户问题进行ES检索,返回N(可配置)个相关的初始候选集;值得一提的是,Redis的作用是对话状态管理,即每一个用户于系统交互都会在Redis中创建一个与之对应的对话状态(dialogue status),这个对话状态可以用来区分不同用户,也可以用来进行多轮对话(保存上一步对话的节点数据)。
对话是核心功能,提供一问一答的交互式方式。
人机对话:用户提出问题,系统给出回答。
FAQ问答对持久化保存在MySQL中,管理员只需维护MySQL中的数据。但是在对话时,系统不会去访问MySQL,而是通过ES检索引擎进行检索。因此,保证MySQL和ES的数据一致非常重要。
全量同步:将MySQL中的问答对数据全部同步到ES索引中。
更新多轮问答树:多轮问答基于多轮JSON,逻辑上为树的组织结构,需要将JSON文件读取到Redis中存起来。
打开浏览器访问http://localhost:1234/faq/swagger-ui/可以查看全部接口并进行测试。

打开ui/dialogue.html进行界面交互。以下显示了单轮对话和多轮对话的简单示例。
(值得一提的,前端ui用的是Alibaba开源的对话框架,感觉非常实用,只需要懂点JS就可以调了。)


整个FAQ问答系统就用了一张表,faq问答对,名称为faq_pair,表结构如下:
| 字段名 | 字段类型 | 是否可为空 | 键 | 注释 |
|---|---|---|---|---|
| id | int(11) | NO | PRI | |
| qa_id | int(11) | NO | UNI | 标准问-标准答的唯一标识id |
| standard_question | text | NO | 标准问,表示高频问题 | |
| standard_answer | text | NO | 标准答,表示高频问题对应的回答 |
该系统配置一些自定义的状态码和说明,用一个枚举类CodeMsg表示。
这些状态码可以用于定位问题所在,也可以让前端区分不同的返回值代表的含义等等。
public enum CodeMsg {
//通用状态码10000系列,模块异常
ELASTICSEARCH_EXCEPTION(10001, "elasticsearch异常"),
MYSQL_EXCEPTION(10002, "mysql异常"),
SIMILARITY_NULL_EXCEPTION(10003, "相似度计算模型异常"),
//通用状态码20000系列,有返回值,无异常
SUCCESS(20000, "success"),
SUCCESS_SINGLE(20001, "success-->单轮"),
SUCCESS_MULTI(20002, "success-->多轮"),
//通用状态码30000系列,中间状态
OPTIONS_NOT_HIT(30001, "处于多轮问答中,但未命中多轮问答的选项,此时将重新检索用户问题"),
//通用状态码40000系列,无返回值
FAILED(40000, "failed"),
UNRECOGNIZED_QUESTION(40001, "failed-->无法识别的问题"),
MULTI_ROUND_QA_NOT_FOUND(40002, "failed-->没有找到对应的多轮问答树"),
MULTI_ROUND_QA_NULL(40003, "failed-->redis中多轮问答树为空"),
MULTI_ROUND_QA_CHILD_NODE_NULL(40004, "failed-->多轮问答树子节点为空");
}
项目定义了用户配置文件application-user.yml,通过在SpringBoot默认配置文件application.yml中配置以下参数引入该配置文件
spring:
#引入自定义配置,application-user.yml
profiles:
include:
- user
application-user.yml中自定义了一些参数,可以根据需要随时修改而不用改源码,如对话相关的参数:
#对话配置
dialogue:
#置信度排序
confidence-rank:
#返回的置信度最高的doc的个数
size: 5
#置信度计算权重
weights:
#相关度权重
relevance-weight: 0.3
#相似度权重
similarity-weight: 0.7
#用户对话状态
status:
#过期时间(单位: minute)
expire-time: 2
#多轮问答树
multi-turn-qa:
path: data/multi_turn_qa
#redis热点数据缓存
hot-data:
#是否开启
open: true
#过期时间(单位: minute)
expire-time: 5
对话相关参数的配置类如下:
@Configuration
@ConfigurationProperties(prefix = "dialogue")
@Data
public class DialogueConfig {
private ConfidenceRank confidenceRank;
private Status status;
private MultiRoundQa multiTurnQa;
private HotData hotData;
//redis中多轮问答树的key前缀
private final String MQATreeKeyPrefix = "MQATreeNode_";
//redis中question映射id的key
private final String MQAQuestion2idKey = "MQA_question2id";
//redis中用户对话状态的key前缀
private final String DialogueStatusKeyPrefix = "dialogue_status_userId_";
//redis中热点数据的question映射id的key
private final String HotDataQuestion2idKey = "hot_data_question2id";
//redis中热点数据的key前缀
private final String HotDataKeyPrefix = "hot_data_";
@Data
public static class ConfidenceRank {
private Integer size;
private Weights weights;
private Float threshold;
@Data
public static class Weights {
private Float relevanceWeight;
private Float similarityWeight;
}
}
@Data
public static class Status {
private Integer expireTime;
}
@Data
public static class MultiRoundQa {
private String path;
}
@Data
public static class HotData {
private Boolean open;
private Integer expireTime;
}
}
用的也是推荐的依赖包,引入的pom依赖如下:
<!-- 更推荐的读取配置文件的处理器-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>
对话流程表示从用户问题输入,到找到答案输出的流程,流程图如下。

多轮对话按规则执行,逻辑上组织为一棵树,示意图如下:

一颗多轮对话树在物理上为一个json文件,在更新多轮对话树时json文件将被转换成数据对象MultiQaTreeNode,然后添加到redis中。
{
"qaId": 3,
"question": "推荐一个景点",
"answer": "好的,请问对景点评分有要求吗",
"childNodes": [
{
"question": "没要求",
"answer": "好的,请问景点票价可接受范围?",
"childNodes": [
{
"question": "免费",
"answer": "附近好多公园呢,比如xxx,今天天气不错,可以去转转。",
"childNodes": []
},
{
"question": "50元以内",
"answer": "这个xx不错,自然风光秀丽,离您也不远。",
"childNodes": []
},
{
"question": "无所谓,不差钱",
"answer": "推荐xxx景点给您呢,该景点绝对符合您的气质",
"childNodes": []
}
]
},
{
"question": "3分以上",
"answer": "可玩的就比较多了,有xx...",
"childNodes": []
},
{
"question": "5分",
"answer": "在xx那有一处5A景区,评分有5分呢,推荐您去玩哈。",
"childNodes": []
}
]
}
MultiQaTreeNode类如下:
public class MultiQaTreeNode implements Serializable {
//对应的qaId,一棵多轮问答树不同层节点的qaId是相同的,都为根节点question所对应的qaId
private Integer qaId;
//当前节点的问题
private String question;
//当前节点的回答
private String answer;
//当前节点的子节点
private List<MultiQaTreeNode> childNodes;
}
系统是简化版的,基本只保留了人机对话功能,问答对也只用了一张表,实际上对于一个高频问题,可以多生成一些与之相似的问题用于扩大搜索范围。另外,如果需要增加问答对,需要对MySQL数据表增加行数据,然后使用同步功能在ES建对应索引即可。
该项目源码如下:
GitHub
Gitee
另外,为了学习微服务相关技术栈,对单体架构进行了升级,而且增加了问答对管理的一些功能。
微服务版地址为:
Gitee
GitHub
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数
我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden
电脑0x0000001A蓝屏错误怎么U盘重装系统教学分享。有用户电脑开机之后遇到了系统蓝屏的情况。系统蓝屏问题很多时候都是系统bug,只有通过重装系统来进行解决。那么蓝屏问题如何通过U盘重装新系统来解决呢?来看看以下的详细操作方法教学吧。 准备工作: 1、U盘一个(尽量使用8G以上的U盘)。 2、一台正常联网可使用的电脑。 3、ghost或ISO系统镜像文件(Win10系统下载_Win10专业版_windows10正式版下载-系统之家)。 4、在本页面下载U盘启动盘制作工具:系统之家U盘启动工具。 U盘启动盘制作步骤: 注意:制作期间,U盘会被格式化,因此U盘中的重要文件请注
华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o
在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList()Obt
C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.
目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称
MIMO技术的优缺点优点通过下面三个增益来总体概括:阵列增益。阵列增益是指由于接收机通过对接收信号的相干合并而活得的平均SNR的提高。在发射机不知道信道信息的情况下,MIMO系统可以获得的阵列增益与接收天线数成正比复用增益。在采用空间复用方案的MIMO系统中,可以获得复用增益,即信道容量成倍增加。信道容量的增加与min(Nt,Nr)成正比分集增益。在采用空间分集方案的MIMO系统中,可以获得分集增益,即可靠性性能的改善。分集增益用独立衰落支路数来描述,即分集指数。在使用了空时编码的MIMO系统中,由于接收天线或发射天线之间的间距较远,可认为它们各自的大尺度衰落是相互独立的,因此分布式MIMO
项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU