Ruby How search and index works (Ruby 语言描述)

hooopo for Shopper+ · 2014年12月28日 · 最后由 memorycancel 回复于 2015年04月01日 · 9926 次阅读
本帖已被管理员设置为精华贴

这篇文章主要目的是简单介绍搜索引擎的检索过程,并通过简单的 Ruby 代码实现。当然,由于本人能力有限,对搜索也是一知半解,注定话题既不会太深,也不会太广...这不是 Solr 配置详解这种过于务实的话题,也不是某个搜索排序算法的深入研究的高深话题。

Tokenize (分词)

Tokenize 的过程是把 Query 或待索引的文本拆成最小的单元,也就是分词。包括以下过程:

  • Preprocess text
    • Synonym substitution
    • Remove illegal expressions
    • Remove stopwords.
  • Splitting and preparations for tokenizing
    • Split the text into words.
    • Truncate the amount of tokens.
  • Creating tokens / strings
    • Downcase each token.
    • Stemming each token. *

Synonym Rings (同义词环) 同义词环(如图),把一组定义为等价关系的词汇连接起来,以提供搜索之用。事实上,这些词通常不是真正的同义词。例如,你在检查搜索日志,并和用户对话之后,你将发现不同的人在寻找同样的东西时,会输入不同的术语。某人要找 food processor(食物搅拌机),可能输入 blender (搅拌机),或下图中任何一个术语(或者常见的错误拼法)。不同类型网站,有着各自领域的同义词。

Stopwords (停词)正如字面意思,停词。有些高频词,例如“a”,对用户查询的意义不大,同时在索引中又占据比较多的资源,最重要的是这些词还会影响匹配文档的相关度,所以,干脆在分词的时候就把它去掉。

Automatic Stemming (自动词干) 功能,是搜索引擎的一个实用功能,把一个术语扩展,包含其他共享相同词干的术语。如果词干搜索机制很强,可能会把 computer 这个搜索术语视为和 computerscomputationcomputing 甚至 comput 这些术语共享相同的词干 comput

其他过程很容易理解了,整个 Tokenize 用 Ruby 代码的简单表示:

def tokenize(text)
  text.gsub(/&/, "and").                      # Synonym substitution
       gsub(/[^\w\s]+/, "").                  # Remove illegal expressions
       gsub(/\b(a|an|is)\b/i, "").            # Remove stopwords
       split(/\s+/).map do |token|            # Split the text into words
         token.downcase.sub(/(.*?)s\Z/, '\1') # Downcase each token
       end
end

tokenize("How search & index works?") #=> ["how", "search", "and", "index", "work"]

其中 Stemming 过程过于简陋,但相关算法其实已经非常成熟了,比如 Porter、Hunspell 等算法。其中 Ruby 实现有:

Inverted Index (倒排索引)

倒排索引(英语:Inverted index),也常被称为反向索引、置入档案或反向档案,是一种索引方法,被用来存储在全文搜索下某个单词在一个文档或者一组文档中的存储位置的映射。它是文档检索系统中最常用的数据结构。

以英文为例,下面是要被索引的文本:

T_0="it is what it is"
T_1="what is it"
T_2="it is a banana"

正向索引

docs = {
  1 => "it is what it is",
  2 => "what is it",
  3 => "it is a banana"
}

倒排索引

"a":      {2}
"banana": {2}
"is":     {0, 1, 2}
"it":     {0, 1, 2}
"what":   {0, 1}

检索的条件"what", "is" 和 "it" 将对应这个集合:{0,1} & {0,1,2} & {0,1,2} = {0,1}。即:0 号和 1 号文档包含 what is it 这三个关键词。

接着上面的 Tokenize,用 Ruby 简单实现倒排索引过程,其中对 Query 分词和对文档分词使用的是同一方式(有时候两者会有微小差异):

require 'set'
require 'pry'

docs = {
  1 => "it is what it is",
  2 => "what is it",
  3 => "it is a banana"
}

def tokenize(text)
  text.gsub(/&/, "and").
       gsub(/[^\w\s]+/, "").
       gsub(/\b(a|an|is)\b/i, "").
       split(/\s+/).map do |token|
         token.downcase.sub(/(.*?)s\Z/, '\1')
       end
end

index = {}

docs.each do |id, doc|
  tokenize(doc).each do |word|
    if index.has_key?(word)
      index[word] << id
    else
      index[word] = Set.new([id])
    end
  end
end

def search(text, index)
  tokenize(text).map{ |word| index[word] || Set.new }.reduce(:&)
end

index_store = init_index_store(Docs)   => {"it"=>#<Set: {1, 2, 3}>, "what"=>#<Set: {1, 2}>, "banana"=>#<Set: {3}>}
search("what is it?", index_store)     => #<Set: {1, 2}>

至此,一个最简单是 search engine 就实现了,当然,我们还没涉及到排序,搜索过程使用的是最简单的 集合布尔模型。下面会涉及 TF-IDF 相关度排序内容。

Sort By Relevance (按相关度排序)

TODO

原文连接:http://shopperplus.github.io/blog/2014/12/28/how-search-and-index-works.html

我看还不如看 Whoosh 的文档和代码,我觉得 Whoosh 挺好的,改起来也挺方便的

https://pythonhosted.org/Whoosh/

hooopo 你写晚了几年

#2 楼 @bhuztez 为何不去看 lucence 代码

这贴暴露了 @hooopo@hooooopo 是一对好基友,两兄弟。

#3 楼 @hooooopo lucence 改起来费劲啊

#4 楼 @lgn21st 因为手机上不能用 github 登录,又不记得密码……

提起搜索引擎不能不说 bloom filter 啊,有 80 亿网页要爬,光 url 判定去重就是一个 工作量,如果用 hash 存储,每个 url 假设 50 字节,就是 80*亿*50byte=50G,嘿嘿, 这也是一个 pratical 的考虑,必须祭出 bloom filter 宝刀了。

匿名 #8 2014年12月28日

:plus1:

写得好

先标记,再补刀

#10 楼 @luikore 嗯,为了简化,没写那么复杂。

fourty-four 和 fourtyfour 等价应该算是一个 WordDelimiter 相关的话题,solr 的文档描述的更全一些: http://wiki.apache.org/solr/AnalyzersTokenizersTokenFilters#solr.WordDelimiterFilterFactory

可能还需要根据驼峰或数字与字母间隔去拆:fourtyFour -> fourty four, windows2000 -> windows 2000

先收藏了。

赞,之前毕设做一个 twitter 个性化搜索的东西,现在看来差不多就是这流程。虽然我是用来提取关键词的,不过基本一致

第二个图裂了

@hooopo 我这里的第二个图也裂了

收藏了 回去慢慢看

小例子不错,理解理解练练手也好啊~

需要 登录 后方可回复, 如果你还没有账号请 注册新账号