MongoDB源码解析:Full Text Search Index

框架实现

FTS本质上也是Btree索引类型

索引AccessMethod定义:

class FTSAccessMethod : public BtreeBasedAccessMethod

关键成员:

fts::FTSSpec _ftsSpec;

获取索引的函数入口:

void FTSAccessMethod::getKeys(const BSONObj& obj, BSONObjSet* keys) {
  ExpressionKeysPrivate::getFTSKeys(obj, _ftsSpec, keys);
}

追踪到:
fts::FTSIndexFormat::getKeys(ftsSpec, obj, keys);

  • ftsSepc:是用来描述语言类型的类,定义的语言索引的一系列属性
  • obj:需要被索引的文档
  • keys:分词后产生的结果

FTSIndexFormat的实现:

  • getKeys:找多文档所有命中的索引,FTS也支持组合索引,全文索引分词的建立通过函数FTSSpec::scoreDocument,注意:单个文档所有的索引加起来不能大于4MB,数量小于400000个。所以MongoDB的全文索引不实用于较大的文档。尤其是中文情况下,由于中文的复杂性,所以要格外注意,从性能角度上建议把文档控制在几百KB左右范围内。
  • getIndexKey:获取索引Key,VERSION2的算法对较长的Key进行了Hash压缩,算法是截取前32字节的前缀,后32字节用Murmurhash运算出来的数值代替,所以,最长不会超过64字节。

FTSSpec::scoreDocument,分词,然后统计频度,最后算法评分,算法原则是单词出现比例越大分数约高,单词的基数越大,造成的分数差距越小,算法大概如下:

  • exp = 2^(count-1)
  • freq = SIGMA(1/exp)
  • coeff = (count + totalCount) / ( 2 * totalCount)
  • score = weight * freq * coeff

分词实现

分词是搜索的基础,MongoDB目前只支持拉丁语系的分词处理,拉丁语的分词相对与中文简单很多,只需要有限的stop-word(比如the,a,空格等)即可完成一个套简单的分词功能。而中文博大精深,目前还没有非常好的开源库实现中文分词。

具体算法,先用stop-word进行文本切割,然后调用libstemmer三方库进行词干提取,最后得到文本分词。

挑战中文分词

经过一天的代码学习,总结下来,要实现中文分词首先要解决的是中文分词器,然后是字库。

有了两者的基础后,可以通过定义VERSION版本来自定一套分词算法,甚至是评分标准。

时间: 2024-11-10 07:59:06

MongoDB源码解析:Full Text Search Index的相关文章

Redux系列x:源码解析

写在前面 redux的源码很简洁,除了applyMiddleware比较绕难以理解外,大部分还是 这里假设读者对redux有一定了解,就不科普redux的概念和API啥的啦,这部分建议直接看官方文档. 此外,源码解析的中文批注版已上传至github,可点击查看.本文相关示例代码,可点击查看. 源码解析概览 将redux下载下来,然后看下他的目录结构. npm install redux 这里我们需要关心的主要是src目录,源码解析需要关心的文件都在这里面了 index.js:redux主文件,主

从源码解析Android中View的容器ViewGroup_Android

 这回我们是深入到ViewGroup内部\,了解ViewGroup的工作,同时会阐述更多有关于View的相关知识.以便为以后能灵活的使用自定义空间打更近一步的基础.希望有志同道合的朋友一起来探讨,深入Android内部,深入理解Android. 一.ViewGroup是什么?       一个ViewGroup是一个可以包含子View的容器,是布局文件和View容器的基类.在这个类里定义了ViewGroup.LayoutParams类,这个类是布局参数的子类.        其实ViewGrou

Step.js 使用教程(附源码解析)

Step.js(https://github.com/creationix/step)是控制流程工具(大小仅 150 行代码),解决回调嵌套层次过多等问题.适用于读文件.查询数据库等回调函数相互依赖,或者分别获取内容最后组合数据返回等应用情景.异步执行简单地可以分为"串行执行"和"并行"执行,下面我们分别去看看. 串行执行 这个库只有一个方法 Step(fns...).Step 方法其参数 fns 允许是多个函数,这些函数被依次执行.Step 利用 this 对象指

Java集合学习(十三) WeakHashMap详细介绍(源码解析)和使用示例

这一章,我们对WeakHashMap进行学习. 我们先对WeakHashMap有个整体认识,然后再学习它的源码,最后再通过实例来学会使用WeakHashMap. 第1部分 WeakHashMap介绍 WeakHashMap简介    WeakHashMap 继承于AbstractMap,实现了Map接口.    和HashMap一样,WeakHashMap 也是一个散列表,它存储的内容也是键值对(key-value)映射,而且键和值都可以是null.   不过WeakHashMap的键是"弱键&

Java集合学习(十一) Hashtable详细介绍(源码解析)和使用示例

这一章,我们对Hashtable进行学习. 我们先对Hashtable有个整体认识,然后再学习它的源码,最后再通过实例来学会使用Hashtable. 第1部分 Hashtable介绍 Hashtable 简介 和HashMap一样,Hashtable 也是一个散列表,它存储的内容是键值对(key-value)映射. Hashtable 继承于Dictionary,实现了Map.Cloneable.java.io.Serializable接口. Hashtable 的函数都是同步的,这意味着它是线

Java集合学习(十) HashMap详细介绍(源码解析)和使用示例

这一章,我们对HashMap进行学习. 我们先对HashMap有个整体认识,然后再学习它的源码,最后再通过实例来学会使用HashMap. 第1部分 HashMap介绍 HashMap简介 HashMap 是一个散列表,它存储的内容是键值对(key-value)映射. HashMap 继承于AbstractMap,实现了Map.Cloneable.java.io.Serializable接口. HashMap 的实现不是同步的,这意味着它不是线程安全的.它的key.value都可以为null.此外

Java集合学习(六) Vector详细介绍(源码解析)和使用示例

学完ArrayList和LinkedList之后,我们接着学习Vector.学习方式还是和之前一样,先对Vector有个整体认识,然后再学习它的源码:最后再通过实例来学会使用它. 第1部分 Vector介绍 Vector简介 Vector 是矢量队列,它是JDK1.0版本添加的类.继承于AbstractList,实现了List, RandomAccess, Cloneable这些接口. Vector 继承了AbstractList,实现了List:所以,它是一个队列,支持相关的添加.删除.修改.

EventBus源码解析

前面一篇文章讲解了EventBus的使用,但是作为开发人员,不能只停留在仅仅会用的层面上,我们还需要弄清楚它的内部实现原理.所以本篇博文将分析EventBus的源码,看看究竟它是如何实现"发布/订阅"功能的. 相关文章EventBus使用详解EventBus源码解析 事件注册 根据前一讲EventBus使用详解我们已经知道EventBus使用首先是需要注册的,注册事件的代码如下: EventBus.getDefault().register(this); EventBus对外提供了一个

JAVA Vector源码解析和示例代码_java

第1部分 Vector介绍Vector 是矢量队列,它是JDK1.0版本添加的类.继承于AbstractList,实现了List, RandomAccess, Cloneable这些接口.Vector 继承了AbstractList,实现了List:所以,它是一个队列,支持相关的添加.删除.修改.遍历等功能.Vector 实现了RandmoAccess接口,即提供了随机访问功能.RandmoAccess是java中用来被List实现,为List提供快速访问功能的.在Vector中,我们即可以通过