[关闭]
@evilking 2018-04-30T02:56:02.000000Z 字数 4803 阅读 1168





SimHash

NLP

SimHash本篇介绍一个文本去重算法,叫SimHash,包括算法原理及java源码分析;最后再简要介绍一下其他相关去重方法.

SimHash原理
简介simhash 是 google 用来处理海量文本去重的算法。simhash最牛逼的一点就是将一个文档,最后转换成一个64位的字节,暂且称之为特征字,然后判断重复只需要判断他们的特征字的距离是不是 < n(根据经验这个n一般取值为3),就可以判断两个文档是否相似。

simhash 本质上是一种LSH,它是局部敏感的;所谓局部敏感是指非常相似的文本,即便只差一个字符,MD5 后的序列也可能非常不同,但是simhash之后的序列可能只是某几位不同;这样对于不同的文本来说,可能就几个字不同,使用simhash就能判断文本是否相似,而传统 MD5或者 hashcode就不行了。

算法描述这里直接给出算法过程的 5 步描述 : 分词,hash,加权,合并,转换.

分词分词是把需要去重的两个文本先通过分词工具分词,然后去掉噪音词,并给每个词加上权重(此数字越大代表这个词对文本越重要,一般可以设置为词频,或者TF-IDF也行).

例如:"美国"51区"雇员称内部有9架飞碟,曾看见灰色外星人",分词之后为 "美国(4) 51区(5) 雇员(3) 称(1) 内部(2) 有(1) 9架(3) 飞碟(5) 曾(1) 看见(3) 灰色(4) 外星人(5)",括号里的数字表示权重.

hash此过程是把上一步分词后的每个单词,通过hash算法转换成 hash值,这里指定hashcode的长度,用来控制hash散列的精度.

比如"美国"通过hash算法计算为"100101","51区"通过hash算法计算为"101011";这样我们的字符串就变成了一串串固定长度的数字,提高了相似度的计算性能.

加权通过上两步的操作结果,需要按照单词的对应位加权,形成加权数字串.

比如 "美国"的hash值为"100101",通过加权计算为"4 -4 -4 4 -4 4";"51区"的hash值为"101011",通过加权计算为"5 -5 5 -5 5 5";

合并把上步各个单词算出来的序列值按对应位进行累加,合并成一个序列串.

比如"美国"的"4 -4 -4 4 -4 4","51区"的"5 -5 5 -5 5 5",把每一对应位进行累加,"4+5 -4+-5 -4+5 4+-5 -4+5 4+5"计算为"9 -9 1 -1 1 9";

这里作为示例只算了两个单词的,实际情况下要算所有单词的序列串累加

转换把上步算出来的序列值"9 -9 1 -1 1 9"转换成 0 1 串,形成我们最终的simhash签名;其中每一位大于0 就记为 1,小于 0 记为 0;

比如上述例子中,"9 -9 1 -1 1 9"就转换为了"1 0 1 0 1 1"

SimHash过程


文档去重为了计算两篇文档是否相似,先通过上述步骤分别计算出各自的 simhash 签名,然后比较两个 simhash 的汉明距离 ( Hamming Distance ),如果 hamming_distance < k,就认为这两篇文档相似,那这个 k 设置为多少比较合理呢?

在距离为 3 时是一个比较折中的点,在距离为 10 时效果已经很差了,不过我们测试短文本很多看起来相似的距离确实为 10;如果使用距离为 3,短文本大量重复信息不会被过滤,如果使用距离为 10,长文本的错误率也非常高。如何选择,需要根据业务来进行调试。

汉明距离阈值

汉明距离是计算两个等长字符串对应位不同的数量.

比如"101011"与"100101"的汉明距离为 3


java 代码分析
代码准备这里使用github上的一个开源代码来讲解simhash去重的实现原理 : https://github.com/sing1ee/simhash-java

导入eclipse后,java工程目录如下:

simhash目录结构

编写测试用例 TestMain类,代码如下:

  1. public static void main(String[] args) {
  2. String test1 = "aaaaaaaaaaaaaaaaaaaa";
  3. String test2 = "aaaaaaaaaajfljaslfjaslddsjflasdjaaaaaz";
  4. //初始化Simhash工具
  5. //这里主要是为了初始化其中的分词工具
  6. Simhash simHash = new Simhash(new BinaryWordSeg());
  7. //计算 64位的simhash
  8. long docHash = simHash.simhash64(test1);
  9. long docHash2 = simHash.simhash64(test2);
  10. //计算两 simhash的汉明距离
  11. int dist = simHash.hammingDistance(docHash, docHash2);
  12. System.out.println(dist);
  13. }
运行结果如下:

  1. 2

源码分析Simhash构造方法中传入对象,这就是一个分词工具的类,读者可以换成任何其他的分词工具.

  1. public Simhash(IWordSeg wordSeg) {
  2. this.wordSeg = wordSeg;
  3. }
由于64位的simhash与32位的simhash只是在生成hashcode指纹的时候使用的位数不一样而已,所以这里笔者只以 simhash64来分析:

  1. public long simhash64(String doc) {
  2. //指定使用64位的hashcode
  3. int bitLen = 64;
  4. int[] bits = new int[bitLen];
  5. //分词
  6. List<String> tokens = wordSeg.tokens(doc);
  7. for (String t : tokens) {
  8. //对每个词计算 hashcode
  9. long v = MurmurHash.hash64(t);
  10. //这里是为了将整型数按位拆分成 0 1 串
  11. //并以词频为权重,按位累加
  12. for (int i = bitLen; i >= 1; --i) {
  13. if (((v >> (bitLen - i)) & 1) == 1)
  14. ++bits[i - 1];
  15. else
  16. --bits[i - 1];
  17. }
  18. }
  19. long hash = 0x0000000000000000;
  20. long one = 0x0000000000000001;
  21. for (int i = bitLen; i >= 1; --i) {
  22. //每位大于0 的就设置为 1
  23. if (bits[i - 1] > 1) {
  24. hash |= one;
  25. }
  26. //
  27. one = one << 1;
  28. }
  29. return hash;
  30. }
这段源码过程基本与上面讲述的 simhash算法过程一致,读者可以根据自己项目情况做修改.


simhash签名计算完后,如果要计算两篇文档是否相似,就需要计算两simhash字符串的汉明距离;

笔者最开始的实现没有使用位运算,处理性能就不是太好,看了这里用位运算处理的源码,处理性能提升很明显.


海量数据优化方案随着业务的增长,每天都有大量的数据产生,如果一天100w,10天就是1000w了,插入一条数据就要去比较1000w次,显然这个计算量是巨大的。

针对海量数据的去重优化,我们可以将 64 位指纹,切分成 4 份 16 位的数据块,根据抽屉原理在汉明距离为 3 的情况下,如果两个文档相似,那么它必有一块的数据时相等的,如图:

simhash优化1

然后将 4 份数据通过 k-v 数据库或倒排索引存储起来,k 为 16 位截断指纹,v 为 k 相等时剩余的 48 位指纹集合,查询的时候,精确匹配这个指纹的 4 个 16位截断,如图所示 :

simhash优化2

如此,假设样本库有 2^{34} 条数据(约171亿数据),假设数据均匀分布,则每个 16 位(16 个 0 1 数字随机组成的组合为 2^{16} 个 )倒排返回的最大数量为 2^{34}/2^{16} = 2^{34 - 16} = 262144 个候选结果,4 个 16 位截断索引,总的结果为 : 4 * 262144 = 1048576 ,约为 100多万,通过这样的降维处理,原来需要比较 171亿次,现在只需要比较 100万次即可,大大提升了计算效率.

这种是使用了倒排索引的方式进行了优化

笔者在想是否可以使用图数据库索引的方式来加快速度,simhash分隔的每一小块就是图数据库中的一个实体


相关的去重方案
百度的去重百度的去重算法最简单,就是直接找出此文章的最长的n句话,做一遍hash签名。n一般取3。 工程实现巨简单,据说准确率和召回率都能到达80%以上。

Shingle算法简单描述一下:先对两个文本进行分词,两个分词集合的交集除以两个集合的并集,得到Jaccard系数,通过判断Jaccard系数是否大于某个数,来判断两个集合是否重复。

Shingle算法

Shingle算法也可作为句子相似度的计算方法

若想了解更多Shingle算法细节,请自行搜索.

Rabin指纹去重假设 X = ([b_1,b_2,\cdots,b_n]) 包含了 n 个二进制符号串,并且 b_1 = 1,那么构造一个 (n-1)阶的多项式 X(f)为:

具体可参考《Rabin指纹去重算法在搜索引擎中的应用》这篇论文.


R 中使用 simhash
  1. # 创建 worker启动器,计算simhash,取前两个关键词
  2. > simhasher = worker("simhash",topn = 2)
  3. > simhasher <= "你妈妈叫你回家吃饭啊"
  4. $simhash</span></code></li><li class="L5"><code class="language-R"><span class="pun">[</span><span class="lit">1</span><span class="pun">]</span><span class="pln"> </span><span class="str">"5255740375710833686"</span></code></li><li class="L6"><code class="language-R"></code></li><li class="L7"><code class="language-R"><span class="pln">$keyword
  5. 6.55531 6.44422
  6. "妈妈" "吃饭"
  7. > simhasher <= "你妈妈喊你回家吃饭,回家喽"
  8. $simhash</span></code></li><li class="L3"><code class="language-R"><span class="pun">[</span><span class="lit">1</span><span class="pun">]</span><span class="pln"> </span><span class="str">"9184284471008831268"</span></code></li><li class="L4"><code class="language-R"></code></li><li class="L5"><code class="language-R"><span class="pln">$keyword
  9. 12.3845 6.55531
  10. "回家" "妈妈"
  11. >
  12. # 计算hamming distance
  13. > distance("你妈妈喊你回家吃饭","你妈妈叫你回家吃饭啊",simhasher)
  14. $distance</span></code></li><li class="L3"><code class="language-R"><span class="pun">[</span><span class="lit">1</span><span class="pun">]</span><span class="pln"> </span><span class="lit">0</span></code></li><li class="L4"><code class="language-R"></code></li><li class="L5"><code class="language-R"><span class="pln">$lhs
  15. 6.55531 6.44422
  16. "妈妈" "吃饭"
  17. $rhs
  18. 6.55531 6.44422
  19. "妈妈" "吃饭"
  20. >


添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注