局部敏感哈希算法

本贴最后更新于 2572 天前,其中的信息可能已经天翻地覆

来源:Poll 的笔记

局部敏感哈希(Locality Sensitive Hashing,LSH)算法是我在前一段时间找工作时接触到的一种衡量文本相似度的算法。局部敏感哈希是近似最近邻搜索算法中最流行的一种,它有坚实的理论依据并且在高维数据空间中表现优异。它的主要作用就是从海量的数据中挖掘出相似的数据,可以具体应用到文本相似度检测、网页搜索等领域。

1. 基本思想

局部敏感哈希的基本思想类似于一种空间域转换思想,LSH 算法基于一个假设,如果两个文本在原有的数据空间是相似的,那么分别经过哈希函数转换以后的****它们也具有很高的相似度;相反,如果它们本身是不相似的,那么经过转换后它们应仍不具有相似性。

**
**

哈希函数,大家一定都很熟悉,那么什么样的哈希函数可以具有上述的功能呢,可以保持数据转化前后的相似性?当然,答案就是局部敏感哈希。

2. 局部敏感哈希 LSH

局部敏感哈希的最大特点就在于保持数据的相似性,我们通过一个反例来具体介绍一下。

假设一个哈希函数为 Hash(x) = x%8,那么我们现在有三个数据分别为 255、257 和 1023,我们知道 255 和 257 本身在数值上具有很小的差距,也就是说它们在三者中比较相似。我们将上述的三个数据通过 Hash 函数转换:

Hash(255) = 255%8 = 7;

Hash(257) = 257%8 = 1;

Hash(1023) = 1023%8 = 7;

我们通过上述的转换结果可以看出,本身很相似的 255 和 257 在转换以后变得差距很大,而在数值上差很多的 255 和 1023 却对应相同的转换结果。从这个例子我们可以看出,上述的 Hash 函数从数值相似度角度来看,它不是一个局部敏感哈希,因为经过它转换后的数据的相似性丧失了。

我们说局部敏感哈希要求能够保持数据的相似性,那么很多人怀疑这样的哈希函数是否真的存在。我们这样去思考这样一个极端的条件,假设一个局部敏感哈希函数具有 10 个不同的输出值,而现在我们具有 11 个完全没有相似度的数据,那么它们经过这个哈希函数必然至少存在两个不相似的数据变为了相似数据。从这个假设中,我们应该意识到局部敏感哈希是相对的,而且我们所说的保持数据的相似度不是说保持 100% 的相似度,而是保持最大可能的相似度

对于局部敏感哈希“保持最大可能的相似度”的这一点,我们也可以从数据降维的角度去考虑。数据对应的维度越高,信息量也就越大,相反,如果数据进行了降维,那么毫无疑问数据所反映的信息必然会有损失。哈希函数从本质上来看就是一直在扮演数据降维的角色。

** 3. 文档相似度计算**

我们通过利用 LSH 来实现文档的相似度计算这个实例来介绍一下 LSH 的具体用法。

3.1 Shingling

假设现在有 4 个网页,我们将它们分别进行 Shingling(将待查询的字符串集进行映射,映射到一个集合里,如字符串“abcdeeee", 映射到集合”(a,b,c,d,e)", 注意集合中元素是无重复的,这一步骤就叫做 Shingling, 意即构建文档中的短字符串集合,即 shingle 集合。),得到如下的特征矩阵:

其中“1”代表对应位置的 Shingles 在文档中出现过,“0”则代表没有出现过。

在衡量文档的相似度中,我们有很多的方法去完成,比如利用欧式距离、编辑距离、余弦距离、Jaccard 距离等来进行相似度的度量。在这里我们运用 Jaccard 相似度。接下来我们就要去找一种哈希函数,使得在 hash 后尽量还能保持这些文档之间的 Jaccard 相似度,即:

我们的目标就是找到这样一种哈希函数,如果原来文档的 Jaccard 相似度高,那么它们的 hash 值相同的概率高,如果原来文档的 Jaccard 相似度低,那么它们的 hash 值不相同的概率高,我们称之为 Min-hashing(最小哈希)。

3.2 Min-hashing

Min-hashing 定义为:特征矩阵按行进行一个随机的排列后,第一个列值为 1 的行的行号。举例说明如下,假设之前的特征矩阵按行进行的一个随机排列如下:

最小哈希值:h(S1)=3,h(S2)=5,h(S3)=1,h(S4)=2.

为什么定义最小 hash?事实上,两列的最小 hash 值就是这两列的 Jaccard 相似度的一个估计,换句话说,两列最小 hash 值同等的概率与其相似度相等,即 P(h(Si)=h(Sj)) = sim(Si,Sj)。为什么会相等?我们考虑 Si 和 Sj 这两列,它们所在的行的所有可能结果可以分成如下三类:

  1. A 类:两列的值都为 1;

  2. B 类:其中一列的值为 0,另一列的值为 1;

  3. C 类:两列的值都为 0.

特征矩阵相当稀疏,导致大部分的行都属于 C 类,但只有 A、B 类行的决定 sim(Si,Sj),假定 A 类行有 a 个,B 类行有 b 个,那么 sim(si,sj)=a/(a+b)。现在我们只需要证明对矩阵行进行随机排列,两个的最小 hash 值相等的概率 P(h(Si)=h(Sj))=a/(a+b),如果我们把 C 类行都删掉,那么第一行不是 A 类行就是 B 类行,如果第一行是 A 类行那么 h(Si)=h(Sj),因此 P(h(Si)=h(Sj))=P(删掉 C 类行后,第一行为 A 类)=A 类行的数目/所有行的数目=a/(a+b),这就是最小 hash 的神奇之处。

Min-hashing 的具体做法可以根据如下进行表述:

返回到我们的实例,我们首先生成一堆随机置换,把特征矩阵的每一行进行置换,然后 hash function 就定义为把一个列 C hash 成一个这样的值:就是在置换后的列 C 上,第一个值为 1 的行的行号。如下图所示:

图中展示了三个置换,就是彩色的那三个,我现在解释其中的一个,另外两个同理。比如现在看蓝色的那个置换,置换后的 Signature Matrix 为:

然后看第一列的第一个是 1 的行是第几行,是第 2 行,同理再看二三四列,分别是 1,2,1,因此这四列(四个 document)在这个置换下,被哈希成了 2,1,2,1,就是右图中的蓝色部分,也就相当于每个 document 现在是 1 维。再通过另外两个置换然后再 hash,又得到右边的另外两行,于是最终结果是每个 document 从 7 维降到了 3 维。我们来看看降维后的相似度情况,就是右下角那个表,给出了降维后的 document 两两之间的相似性。可以看出,还是挺准确的,回想一下刚刚说的:希望原来 documents 的 Jaccard 相似度高,那么它们的 hash 值相同的概率高,如果原来 documents 的 Jaccard 相似度低,那么它们的 hash 值不相同的概率高,如何进行概率上的保证?Min-Hashing 有个惊人的性质:

就是说,对于两个 document,在 Min-Hashing 方法中,它们 hash 值相等的概率等于它们降维前的 Jaccard 相似度。

注:在上述例子中,我们还可以采取欧氏距离等相似度量来替代 Jaccard 相似度,这时候 LSH 相应的策略也需要进行改变,从而使得最后的 hash 值等于降为前的相似度。

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...