【迅搜03】全文检索、文档、倒排索引与分词

news2025/1/24 20:27:43

全文检索、文档、倒排索引与分词

今天还是概念性的内容,但是这些概念却是整个搜索引擎中最重要的概念。可以说,所有的搜索引擎就是实现了类似的概念才能称之为搜索引擎。而且今天的内容其实都是相关联的,所以不要以为标题上有四个名词就感觉好像内容很多一样,其实它们都是联系紧密的,一环套一环的。

全文检索

先来看看啥叫 全文检索 。

全文检索是指计算机索引程序通过扫描文章中的每一个词,对每一个词建立一个索引,指明该词在文章中出现的次数和位置,当用户查询时,检索程序就根据事先建立的索引进行查找,并将查找的结果反馈给用户的检索方式。这个过程类似于通过字典中的检索字表查字的过程。全文搜索搜索引擎数据库中的数据。

又是百科上的定义。但是,不管是 XS 还是 ES ,我们有时也都会叫它们为 全文检索引擎 。现在知道了吧,搜索引擎组件,最核心的功能其实就在于全文检索的能力。而在全文检索的过程中,最核心的又是索引的建立,在上面概念中那句:通过字典中的检索字表查字的过程。就是对于索引这个概念的一般性描述。

在学习 ES 的过程中,会提到两个概念,我觉得这两个概念也是非常重要的。那就是 精确值 和 全文本 。

  • 精确值:那些不需要建立索引、不需要分词的确定值,比如说 ID、时间、数字,也就是我们在 MySQL 中不会用到 Like 的字段。

  • 全文本:需要分词,需要对内容建立索引的值,比如文章标题、关键字、内容、描述等,一般都是 Text 类型,是我们搜索时主要面对的内容,也是我们在 MySQL 中需要进行 Like 查询的字段。

有了这两个概念之后,以后我们在建立索引时,就可以根据字段的不同情况,来决定字段是否应该分词建索引。如果不管任何字段都建立索引的话,会导致索引巨大,带来额外的存储及性能开销。

在全文检索中,还有两个概念,那就是 按字检索 和 按词检索 。

对于英文来说,搜索分词其实并不困难。因为英文在书写时天生就有空格分隔。即使是一些简写,比如 I'm ,也可以直接划分为整个词,或者做成 I am 的替换词或近意词。但中文就不一样了。就像这一段话,你觉得要怎么切分呢?能切出来多少词呢?用程序怎么实现?

这就是中文分词的难点。如果说 按字检索 ,就会导致索引巨大。每一个字,每一个字和后面的句子的连接,都要建立索引。比如“我爱北京天安门”,如果按字分词检索,需要切分成:

看着还好是吧?但是换成一整篇文章呢?再换成上百万篇文章呢?

那么按词分词检索呢?我们就可以把上面那句话切分成:

  • 我爱

  • 北京

  • 天安门

先不说查询,至少存储空间就能节约不少吧。后面我们马上就会详细地说分词这件事。

文档

文档在搜索引擎中,就是实际的存储的数据单元。说直白点,就是我们在 MySQL 的那一行数据,将一行数据放到搜索引擎中,就是一篇文档。只不过这个文档是结构化的,有结构属性的,有字段名和值,可以被查询检索出来的数据。在搜索引擎中,文档是一个重要的概念,我们增、删、改、查操作的都是文档。而文档在进行上述操作时,又会关联到索引的建立。

在 XS 中,使用 PHP SDK 时,专门的 XSDocument 是贯穿我们学习始终的一个对象。同样地,在 ES 中,所有数据只有一个 type 类型,就是 _doc 类型。这个不用多解释了吧,doc 就是文档的意思。

或者再换句话说,我们上面所说的全文检索引擎,以及我们这个系列要学习的搜索引擎,这两个概念,最终都会落在 文档搜索引擎 这个概念上。因为不管字段属性如何,我们其实一直都是在搜索文档。

倒排索引

到我们的重点概念咯,这也是常见的面试题。那就是啥是倒排索引?

要理解倒排索引(反向索引),我们就先要了解一下正排索引(正向索引)是啥。这个其实不用多说了吧,普通关系型数据库的索引都是正排索引嘛。我们为某个字段建立索引,然后形成一颗 B+树 。大概长这样。

8a26c940ce020e4ee1cdfa2bc5b91f47.jpeg

典型的一个 MySQL InnoDB B+树索引图,数据有三个字段,id、name、age ,然后 id 上有主键索引,name 和 age 做了联合索引。在关系型数据库中,索引上存储的是字段具体的值,然后索引根据这些值排序。这样在搜索查找时就可以利用类似于二分查找的方式快速找到与查找值匹配的索引项目。如果是普通索引,则会找到最终的主键 ID ,进而再使用主键 ID 通过主键索引(聚集索引)返表查询出实际的数据。

这一套内容大家在学习数据库相关知识的时候应该都会看到,这个不是我们的重点内容,毕竟咱们现在是要讲搜索引擎呀。那么搜索引擎里的倒排索引是啥意思呢?还是看图来说。

8b8270880d9cf114642f8d54ff4c7d54.jpeg

不得不说,倒排索引的图还好画一些。在这个图中,我们假设全文本字段为 subject 和 message ,那么我们就需要对这两个字段中的内容进行分词,根据分词结果形成一个倒排表。各位大佬一眼就明白了吧,每个词项对应记录的就是这个词所在的文档的 ID 。当然,实际上的倒排索引内容可能不止这两个字段,还会包括关键字在文档中的位置等信息。这里就是简单地以最核心的单词和文档的关系来讲解。

是的,这就是倒排索引。其实最终,它获得的结果和 B+树 的普通索引是类似的,最终都是保存着一份主键 ID ,但 B+树 索引的值是整个表行字段的值,最终记录是在所有分枝之后的一个叶子节点上,而且只有一个值。而倒排索引保存的值是一个一个的词项,相同词项只会有一份,最终记录是一组 ID 。

要说一个比较实际点的例子的话,请唱出《孤勇者》的三句歌词和请说出歌词中包含包含“我爱你”的三首歌名,这种“根据题目查找内容”和“根据关键字查找题目”的形式其实是完全相反的,分别代表的就是正排索引(整个字段建立索引,就像是个题目)和倒排索引(将内容分成单词变成一个字典,通过字典查找内容和题目)。(极客时间:检索核心技术20讲,关于倒排索引的解释)

如果数据非常多,而且也都是大篇文章,那么其实这个词项列表的内容也不少。这时候,大部分搜索引擎其实会在词项上再次运用 B+树 ,也就是通过二分法能够快速定位词项。这样,就可以最快(最理想状态下)通过 O(logn) + O(1) 的速度定位到包含某一个词的全部文档 ID 。

如果我们同时搜索多个关键词,则会在获得所有关键词对应的文档 ID 后,再进行归并或多路归并排序的方法遍历两个单词中所有的文档 ID 所对应的内容,从而达到 O(m+n) 的速度,这里的 m 和 n 指的是单词对应文档 ID 列表,而不是正排索引中的全文档搜索的 n 。因此,它的效率还是可以接受的。(极客时间:检索核心技术20讲及百度查询相关资料)具体的算法原理已经不是我能达到的水平的,各位感兴趣的大佬们还是自己再查找资料进行深入的学习吧。

那么如果是传统关系型数据库的 Like 呢?首先,如果是大型的文章内容,索引会非常大,空间极度浪费。其次,如果是前后都模糊,也就是 like '%项目%' 这种,索引会失效。因为我们要需要逐个文字比较,然后再逐个文档匹配,性能可想而知。

再换个角度用生活中的例子来理解,最典型的就是图书目录。我们最常见的那个图书目录,就是根据章节标题的一个正排索引。而有的书,特别是一些专业型的大型经典图书,就是特别厚的那种。往往在这些书的最后还会有一些词汇表,有的词汇表不仅有词汇的解释,还有这些词汇在哪些章节出现过,甚至是具体的页号。这种带章节号或者页号的词汇表就是倒排索引。比如下面这本计算机经典黑皮书《数据库系统概念》中最后的索引部分的内容。加微信或者公众号,可以找我要到很多电子书哦,其中就包括这本书。

63c7aaae01919bc8a0022a5419c0fd8b.png

好了,从倒排索引这里,我们可以看到 分词 真的是对倒排索引非常重要的一个概念,那么我们就再来理解一下什么叫分词。

分词

顾名思义,分词,就是将一句话,一个段落或者一整篇文章中的单词分解出来。对应到我们的数据中,其实就是将文档中,需要全文本分析的字段内容进行分词处理,然后将获取到的分词加入到倒排索引表。

前面我们就说过,英文分词会相对来说简单一些,中文分词要复杂一些。因此,我们平常不管是使用 ES ,还是 Solr、Sphinx 之类的搜索引擎,都需要同时配置一个中文分词器。现在最流行的,一个是 Jieba ,Python 领域非常出名也是现在整个开发领域都比较流行的分词器。另一个是 IK ,这个是最经典的配合 ES 的分词器,当然,Jieba 也有 ES 的插件。

虽说这两个现在很流行,很出名,但咱们的 XS 所使用的 SCWS ,则是还没有它们的时候就已经存在的,通过 C/C++ 开发的一款优秀分词器。最早,甚至 ES 还没发布的时候,大概 2011 年左右(ES最早是2010年发布的)。我所在的公司使用 Solr 作为搜索引擎,而当时配合 Solr 的分词器就是 SCWS 。

还有更早的,我刚毕业时做过一年 C# ,也就是 ASP.NET 开发,大概是  2009 年。当时的公司使用的是 Lucene.NET 配合 Pangu 分词。

可以看到,不管你是用什么搜索引擎,要在中文世界里使用,分词器都是必不可少的。同时,分词器也不是你理解的就是切分个单词就完了,它还要考虑词性,比如动、名、形容词。这些词性又会影响到词频和逆文档词频分数以及是否索引。再比如助词、语气词:嗯、啊、哈,这些,大部分分词工具直接就会给过滤掉,因为它们对于语义搜索没啥用。

怎么又来一堆名词了?还能不能愉快的玩耍了?

别急别急,这些内容我们后面就可以贯穿在与之相关的内容中一起学习了。现在先有个大概的了解就好了。

所幸,上面的那些分词器,Jieba、IK,以及我们主要要学习的 SCWS ,在形式、功能上都非常相似。就和搜索引擎一样,不管是 ES 还是 XS ,最终都是要实现全文检索的,也要做倒排索引的。因此,学完 XS 的分词以及 SCWS 的分词相关内容后,再看 Jieba 或者 IK 都能很快上手的。

另外,在扩展部分,我们会简单地学习使用 TNTSearch + PHP JIEBA ,到时就会告诉大家,语言实现都是其次的,核心的概念原理才是最重要的,PHP 一样可以实现完整的、纯 PHP 的 搜索引擎+分词 系统,从而实现纯 PHP 的搜索引擎解决方案。

学到这里,不得不提一句。码农们都喜欢抬G贬B。这里的 G 和 B 就不用多做解释了吧,其实 Baidu 的中文搜索很强的,李彦宏大佬“超链分析”相关的论文也对 G 站 PR(PageRank)有重要的影响。中文分词与语义处理,从技术角度和搜索结果来说,Baidu 确实是比 G 站强的。只不过满屏广告影响用户体验,而且码农们喜欢直接复制英文报错信息去 G 站搜索,获得的内容更全面(你懂得),从而让大家觉得 G 站更好。

那么 Baidu 这些大公司,使用的分词器、搜索框架,是我们常见的这些吗?这个我也不知道,但我知道核心原理和概念与我们今天学习到的这些内容都不会相差太远。假如有幸有 Baidu 的大佬看到这篇文章,或者你对这一块有更深入的了解,也欢迎评论区留言赐教哦!

与关系型数据库的对比

最后,我们再来根据今天的知识,将学到的内容与关系型数据库中的相关名词做一个对比,方便大家的记忆。

关系型数据库XSES
表 Table索引 Index索引 Index
行 Row文档 Document文档 Document
列 Column字段 Field字段 Field
表结构 Schema索引配置 ini 文件,XSFieldScheme对象字段映射 mapping
查询语言 SQL 语法无(只能通过 SDK)DSL+RESTFul+Json语法
索引 B+TreeB+Tree、倒排索引B+Tree、倒排索引、分布式

总结

通过今天的学习,相信大家就能够理解为啥上篇文章中使用“项”这个单字无法搜索到结果了吧。不管是 XS 的 SCWS 还是 ES 的 IK ,都不会将“项”作为一个单词拆分出来加入到倒排表中。如果要实现可以索引这个单字的话,那么就需要做成单字倒排索引。试想百万级的文章量,每个文章又是少则几百、多则几万字的内容,那么这个索引量有多恐怖。

当然,如果你只是一些内容很少的,重复度很高的电商产品、订单搜索、内部管理系统OA、HR、项目管理系统之类的,倒是可以尝试单字索引。但通常来说,并不推荐。还是那句话,掌握搜索引擎的概念,我们是做分词语义搜索的,不是 Like 。

具体的分词算法不是我们学习的主要内容,而且这部分内容也非常高级和复杂,包括 NPL 其实也是在加上了 AI 相关的技术之后做得更强大的语义分词以及语法词法分析,也就是说,我们要学习到的以及常用的这些开源分词器,大部分都只是基于普通算法+字典组合的分词器,可以算是 NPL 中的一小部分内容。有兴趣的小伙伴可以自行查阅和学习更深入的资料。

今天的内容是基于自己的学习与理解,参考资料非常多就不一一列举了,如有纰漏希望您能抽出宝贵时间随时批评指正。

下篇文章开始,我们正式进入 XS 的系统与开发学习。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1250142.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

NX二次开发UF_CURVE_ask_int_parms_sc 函数介绍

文章作者:里海 来源网站:https://blog.csdn.net/WangPaiFeiXingYuan UF_CURVE_ask_int_parms_sc Defined in: uf_curve.h int UF_CURVE_ask_int_parms_sc(tag_t int_curve_object, int * num_objects_set_1, tag_t * * object_set_1, int * num_object…

LeetCode Hot100 102.二叉树的层序遍历

题目&#xff1a; 给你二叉树的根节点 root &#xff0c;返回其节点值的 层序遍历 。 &#xff08;即逐层地&#xff0c;从左到右访问所有节点&#xff09;。 方法&#xff1a;迭代 class Solution {public List<List<Integer>> levelOrder(TreeNode root) {if …

基于SpringBoot+Redis的前后端分离外卖项目-苍穹外卖(七)

分页查询、删除和修改菜品 1. 菜品分页查询1.1 需求分析和设计1.1.1 产品原型1.1.2 接口设计 1.2 代码开发1.2.1 设计DTO类1.2.2 设计VO类1.2.3 Controller层1.2.4 Service层接口1.2.5 Service层实现类1.2.6 Mapper层 1.3 功能测试1.3.2 前后端联调测试 2. 删除菜品2.1 需求分析…

局域网协议:DNS(Domain Name System,域名系统)详解

文章目录 什么是DNS&#xff1f;DNS的重要性DNS的域名解析过程递归查询迭代查询 DNS解析失败怎么办&#xff1f;为什么DNS需要递归服务器&#xff1f;DNS为什么用UDP&#xff1f;推荐阅读 什么是DNS&#xff1f; DNS&#xff08;Domain Name System&#xff0c;域名系统&#…

【Unity3D】MAX聚合广告SDK——Pangle广告接入

Pangle, App Monetization Simplified 注册 登录 创建应用 创建广告单元 将其应用ID和广告ID关联到MAX广告。 下载Pangle Unity Plugin包&#xff0c;新建一个空工程&#xff08;很重要&#xff09; Unity版本2019.4.0f1 gradle plugin 4.2.0 gradle版本6.7.1 build_tools 34.…

问题汇总20231124

文章目录 1.练习题错题笔记&#xff1a;2. 串口配置中的无硬件控制流是什么3. 硬件控制流是如何管理数据流的&#xff1f;4. 串口不显示的原因有哪些&#xff1f;5. 中断服务函数中为什么一定要清除中断标志位&#xff1f;6. 中断标志位是什么时候设置的&#xff1f;7. BSRR8. …

探索优雅的处理 JavaScript 类数组对象的技巧

一. 引言 在 JavaScript 编程中&#xff0c;我们经常遇到类数组对象&#xff0c;它们拥有类似数组的结构和行为&#xff0c;但却不具备真正的数组方法和属性。常见的类数组对象包括 DOM 集合、函数的 arguments 对象和字符串等。如果我们想对这些类数组对象进行操作和处理&…

使用 STM32F7 和 TensorFlow Lite 开发低功耗人脸识别设备

本文旨在介绍如何使用 STM32F7 和 TensorFlow Lite框架开发低功耗的人脸识别设备。首先&#xff0c;我们将简要介绍 STM32F7 的特点和能力。接下来&#xff0c;我们将讨论如何使用 TensorFlow Lite 在 STM32F7 上实现人脸识别算法。然后&#xff0c;我们将重点关注如何优化系统…

基于SpringBoot+Redis的前后端分离外卖项目-苍穹外卖(八)

套餐模块功能开发 1. 新增套餐1.1 需求分析和设计1.1.1产品原型&#xff1a;1.1.2接口设计&#xff1a;1.1.3数据库设计&#xff1a; 1.2 代码开发1.2.1 DishController层1.2.2 DishService接口类1.2.3 DishServiceImpl接口实现类1.2.4 DishMapper层1.2.5 DishMapper.xml1.2.6 …

元宇宙现已开放!

在 2023 年 11 月 3 日 The Sandbox 首个全球创作者日上&#xff0c;The Sandbox 联合创始人 Arthur Madrid 和 Sebastien Borget 宣布元宇宙已开放&#xff0c;已创作完整体验的 LAND 持有者可以自行将体验发布至 The Sandbox 地图上。 精选速览 LAND 持有者&#xff1a;如果…

实验室信息管理系统源码,LIS系统源码,lis源码

医学检验(LIS)管理系统源码&#xff0c;云LIS系统全套商业源码 随着全自动生化分析仪、全自动免疫分析仪和全自动血球计数器等仪器的使用&#xff0c;检验科的大多数项目实现了全自动化分析。全自动化分析引入后&#xff0c;组合化验增多&#xff0c;更好的满足了临床需要&…

Android Frameworks 开发总结之七

1.修改android 系统/system/下面文件时权限不够问题 下面提到的方式目前在Bobcat的userdebug image上测试可行&#xff0c;还没有在user上测试过. 修改前: leifleif:~$ adb root restarting adbd as root leifleif:~$ adb disable-verity verity is already disabled using …

集「才华」与「美貌」于一身的原型设计利器—摹客RP

文章目录 画原型做设计&#xff0c;用摹客RP就够了 初遇摹客再遇摹客RP摹客RP简介与注册摹客RP的突出亮点1️⃣拥有海量矢量图标&#xff0c;满足各种设计场景2️⃣打造高扩展性组件&#xff0c;打破传统组件编辑模式3️⃣海量摹客RP模板例子随意挑选4️⃣实现多人实时协同&…

【数据结构】什么是栈?

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:数据结构 ⚙️操作环境:Visual Studio 2022 目录 &#x1f4cc;栈的定义 &#x1f4cc;元素进栈出栈的顺序 &#x1f4cc;栈的抽象数据类型 &#x1f4cc;栈的顺序存储结构 &#x1f4cc;栈的链式存储结构 链栈的进…

micro_ros需要用到的hardware

我没有那么长的线啊&#xff0c;所以就用一个4块5的usb转串口看看 没有那么高档的开发板&#xff0c;就用主流的STM32F103C8T6试试看 这应该就是个仿真器了&#xff0c;一个字不认得都能够看的出来吧

《尚品甄选》:后台系统——权限管理之角色管理(debug一遍)

文章目录 一、权限管理介绍二、表结构的设计三、查询角色四、添加角色五、修改角色六、删除角色 一、权限管理介绍 在后台管理系统中&#xff0c;权限管理是指为了保证系统操作的安全性和可控性&#xff0c;对用户的操作权限进行限制和管理。简单的来说就是某一个用户可以使用…

【开源】基于JAVA的计算机机房作业管理系统

项目编号&#xff1a; S 017 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S017&#xff0c;文末获取源码。} 项目编号&#xff1a;S017&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 登录注册模块2.2 课程管理模块2.3 课…

【开源】基于Vue和SpringBoot的食品生产管理系统

项目编号&#xff1a; S 044 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S044&#xff0c;文末获取源码。} 项目编号&#xff1a;S044&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 加工厂管理模块2.2 客户管理模块2.3…

王道p150 14.假设二叉树采用二叉链表存储结构,设计一个算法,求非空二叉树 b的宽度(即具有结点数最多的那一层的结点个数) (c语言代码实现)

采用层次遍历的方法求出所有结点的层次&#xff0c;并将所有结点和对应的层次放在一个队列中。然后通过扫描队列求出各层的结点总数&#xff0c;最大的层结点总数即为二叉树的宽度。 /* A B C D E F …

使用Pytorch从零开始构建Conditional PixelCNN

条件 PixelCNN PixelCNN 是 PixelRNN 的卷积版本&#xff0c;它将图像中的像素视为一个序列&#xff0c;并在看到前面的像素后预测每个像素&#xff08;定义如上和左&#xff0c;尽管这是任意的&#xff09;。PixelRNN 是图像联合先验分布的自回归模型&#xff1a; p ( x ) …