标签:
这篇文章主要讲simHash算法。这是一种LSH(Locality-Sensitive Hashing,局部敏感哈希)的简单实现。它是广泛用于数据去重的算法,可以用于相似网站、图片的检索。而且当两个样本差别并不大时,算法仍能起效。值得一提的是,该算法的时空复杂度不存在与维度有关的项,所以不会遭遇维度灾难,也可以在维数较高时优化kNN算法。
此算法(LSH)具有双重性,它们似乎是相悖的:
simHash是LSH的其中一种对于字符串的简单实现。操作步骤如下:
这样就可以得出一个字符串的simHash值,时间复杂度为O(|s|)。
定义两个字符串相似,即|hammingDist(simHash(str1),simHash(str2))|<=k,k是最大容忍的不同位数,hammingDist为计算两个整数海明距离的函数。海明距离即为两个整数二进制中编码不同的位数。
根据经验,k一般取3。而海明距离的计算有一种快速的方法,给出C的实现。这种统计二进制中1的个数的算法叫平行算法,本文不再详述。
static int bitCount(unsigned int n){ n=(n &0x55555555)+((n>>1)&0x55555555); n=(n &0x33333333)+((n>>2)&0x33333333); n=(n &0x0f0f0f0f)+((n>>4)&0x0f0f0f0f); n=(n &0x00ff00ff)+((n>>8)&0x00ff00ff); n=(n &0x0000ffff)+((n>>16)&0x0000ffff); return n; } int hammingDist(unsigned int a,unsigned int b){ return bitCount(a^b); }
查找新元素与已知集元素是否相似有两种方法。
光算出simHash值并没有太大的作用,因为判断新元素与已知集的中元素是否相似仍需较长的时间。尤其是数据量很大的时候。这时可以用一定的预处理算法优化第一种算法。
假设k=3。优化的方法如下,将32bit或64bit(下文以32bit为例)的hash值平均分为4段。根据抽屉原理,两个字符串的hash中必有1段中没有不同的位。于是可以将每个元素hash的4个8bit作为键均预存储到表中,值为hash的完整值。查找时,只需比较新字符串hash的4个8bit表中的所有完整hash并判断海明距离是否小于等于3。这样优化后,时间复杂度降至O(4k)=O(4*n/(2^9-1))≈O(n/128),虽然仍为线性复杂度,但已经快了不少。
整个simHash系统的实现(C++版本)我已开源至github:https://github.com/Darksun2010/MLlearning/tree/master/LSH
标签:
原文地址:http://www.cnblogs.com/Darksun/p/5196405.html