MinHash 原理

最小哈希原理介绍

  1. MinHash是基于Jaccard Index相似度(海量数据不可行)的算法,一种降维的方法A,B 两个集合:A = {s1, s3, s6, s8, s9}  B = {s3, s4, s7, s8, s10}
  2. MinHash的基本原理:在A∪B这个大的随机域里,选中的元素落在A∩B这个区域的概率,这个概率就等于Jaccard的相似度

最小哈希:

 

S1

S2

S3

A

1

0

0

B

0

1

0

C

0

0

0

D

1

0

1

行的随机排列转换(也称置换运算)

 

S1

S2

S3

B

0

1

0

D

1

0

1

A

1

0

0

C

0

0

0

哈希值:排列转换后的行排列次序下第一个列值为1的行的行号,例如h(S1)=D,h(S2)=B

两个集合经随机排列之后得到的两个最小哈希值相等的概率等于这两个集合的Jaccard相似度。

问题:

对于上百万甚至数十亿的行选择一个随机排列转换极其消耗时间,怎么办?

  1. 随机选择n个哈希函数h1,h2…
  2. 对每列C进行如下操作:

    a)  如果c在第r行为0,则什么也不做;

    b)   否则,如果c在第r行为1,那么对于每个i=1,2,…,n,将SIG(i,c)置为原来的SIG(i,c)和h(r)中的较小值

 

S1

S2

S3

S4

x+1 mod 5

3x+1 mod 5

0

1

0

0

1

1

1

1

0

0

1

0

2

4

2

0

1

0

1

3

2

3

1

0

1

1

4

0

4

0

0

1

0

0

3

计算最小哈希签名矩阵:

 

S1

S2

S3

S4

h1

1

3

0

1

h2

0

2

0

0

计算Jaccard相似度:

SIM(S1,S4)=1.0;SIM(S1,S3)=1/3;SIM(S1,S2)=0

真实SIM(S1,S4)=2/3;SIM(S1,S3)=1/4;SIM(S1,S2)=0

 

相关论文:

1.《Google News Personalization: Scalable Online Collaborative Filtering》

根据用户的历史点击数据,进行新闻推荐;采用最小哈希聚类的协同过滤算法

2.《The Link-Prediction Problem for Social Networks》

比较社交网络链接预测问题的各种算法

计算好友相似度的流程:

  • 找到N个哈希函数,对每个用户好友集合生成一组Minhash(N个)
  • 对于一个用户,按Minhash相同个数做排序,给出推荐候选集
  • 计算用户跟被备选集的Jaccard Index
  • 按Jaccard结果排序,给出TopN进行推荐

哈希函数生成和哈希值计算:

    1. 输入向量Vector转换为bytes
      • numHashFunctions--预设生成hash函数的个数(假定为10个)
      • for (int i = 0; i < numHashFunctions; i++) {
              for (Vector.Element ele : featureVector) {
                int value = (int) ele.get();
                bytesToHash[0] = (byte) (value >> 24);
                bytesToHash[1] = (byte) (value >> 16);
                bytesToHash[2] = (byte) (value >> 8);
                bytesToHash[3] = (byte) value;
                int hashIndex = hashFunction[i].hash(bytesToHash); //计算哈希函数值
                //只保留最小哈希值
                if (minHashValues[i] > hashIndex) {
                  minHashValues[i] = hashIndex;
                }
              }
            }

         

    2. 采用Mersenne Twister算法构造伪随机生成器
      • Random random = new MersenneTwisterRNG(new FastRandomSeedGenerator());
      • Mersenne Twister(马特赛特旋转演算法),是伪随机数发生器之一,其主要作用是生成伪随机数。
        Mersenne Twister算法的原理:Mersenne Twister算法是利用线性反馈移位寄存器(LFSR)产生随机数的,LFSR的反馈函数是寄存器中某些位的简单异或,这些位也称之为抽头序列。一个n位的LFSR能够在重复之前产生2^n-1位长的伪随机序列。只有具有一定抽头序列的LFSR才能通过所有2^n-1个内部状态,产生2^n - 1位长的伪随机序列,这个输出的序列就称之为m序列。为了使LFSR成为最大周期的LFSR,由抽头序列加上常数1形成的多项式必须是本原多项式。一个n阶本原多项式是不可约多项式,它能整除x^(2*n-1)+1而不能整除x^d+1,其中d能整除2^n-1。例如(32,7,5,3,2,1,0)是指本原多项式x^32+x^7+x^5+x^3+x^2+x+1,把它转化为最大周期LFSR就是在LFSR第32,7,5,2,1位抽头。利用上述两种方法产生周期为m的伪随机序列后,只需要将产生的伪随机序列除以序列的周期,就可以得到(0,1)上均匀分布的伪随机序列了。
        Mersenne Twister有以下优点:随机性好,在计算机上容易实现,占用内存较少(mt19937的C程式码执行仅需624个字的工作区域),产生随机数的速度快、周期长,可达到2^19937-1,且具有623维均匀分布的性质,对于一般的应用来说,足够大了,序列关联比较小,能通过很多随机性测试。
    3. 四种哈希函数生成器
      • MAX_INT_SMALLER_TWIN_PRIME = 2147482949;为什么选这个值?
        • 它是整型范围内最大孪生素数(相差为2的两个数都是质数的情况)的较小值;哈希用素数取模冲突小
      • seedA、seedB、seedC是采用MersenneTwisterRNG随机生成器生成的0~11均匀分布的随机数
      • 第一种:LinearHash

            @Override
            public int hash(byte[] bytes) {
              long hashValue = 31;
              for (long byteVal : bytes) {
                hashValue *= seedA * byteVal;
                hashValue += seedB;
              }
              return Math.abs((int) (hashValue % RandomUtils.MAX_INT_SMALLER_TWIN_PRIME));
            }

      • 第二种:PolynomialHash

         @Override
            public int hash(byte[] bytes) {
              long hashValue = 31;
              for (long byteVal : bytes) {
                hashValue *= seedA * (byteVal >> 4);
                hashValue += seedB * byteVal + seedC;
              }
              return Math.abs((int) (hashValue % RandomUtils.MAX_INT_SMALLER_TWIN_PRIME));
            }

      • 第三种:MurmurHashWrapper
        @Override
            public int hash(byte[] bytes) {
              long hashValue = MurmurHash.hash64A(bytes, seed);
              return Math.abs((int) (hashValue % RandomUtils.MAX_INT_SMALLER_TWIN_PRIME));
            }
      • 第四种:MurmurHash3Wrapper
        @Override
            public int hash(byte[] bytes) {
              long hashValue = MurmurHash3.murmurhash3_x86_32(bytes, 0, bytes.length, seed);
              return Math.abs((int) (hashValue % RandomUtils.MAX_INT_SMALLER_TWIN_PRIME));
            } 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
MinHash是一种用于近似集合相似度计算的技术。下面是一个用Python实现MinHash的示例代码: ```python import numpy as np import hashlib class MinHash: def __init__(self, num_perm): self.num_perm = num_perm self.permutations = self._generate_permutations() def _generate_permutations(self): np.random.seed(0) minhash_permutations = np.random.randint(low=0, high=np.iinfo(np.int64).max, size=(self.num_perm, 2), dtype=np.int64) return minhash_permutations def _hash_value(self, value): return hashlib.sha1(value.encode()).hexdigest() def compute_hash(self, value): hash_value = self._hash_value(value) hash_code = int(hash_value, 16) return hash_code def compute_signature(self, document): signature = np.inf * np.ones(self.num_perm, dtype=np.int64) for word in document.split(): hash_code = self.compute_hash(word) for i in range(self.num_perm): a, b = self.permutations[i] hash_value = (a * hash_code + b) % np.iinfo(np.int64).max signature[i] = min(signature[i], hash_value) return signature def compute_similarity(self, signature1, signature2): return np.mean(signature1 == signature2) # 示例用法 document1 = "This is a document about cats" document2 = "This is a document about dogs" minhash = MinHash(num_perm=128) signature1 = minhash.compute_signature(document1) signature2 = minhash.compute_signature(document2) similarity = minhash.compute_similarity(signature1, signature2) print(f"Similarity between the documents: {similarity}") ``` 在上述示例代码中,我们首先定义了一个MinHash类,它接受参数`num_perm`,表示要使用的哈希函数数量。在初始化时,我们生成了一组随机排列用于哈希计算。 `_hash_value`方法使用SHA1算法对输入值进行哈希计算,并返回哈希值的十六进制表示。 `compute_hash`方法将字符串值转换为哈希码。 `compute_signature`方法计算给定文档的MinHash签名。对于文档中的每个词,我们计算其哈希值,并将其与每个哈希函数的参数相乘并取模。然后,我们将每个哈希函数的最小值作为文档的签名。 `compute_similarity`方法计算两个文档的相似度。它简单地计算两个签名之间相等哈希函数的比例。 在示例用法中,我们创建了两个文档,并使用MinHash计算它们的签名。然后,我们计算了两个签名之间的相似度,并打印了结果。 请注意,此处的示例代码是简化版的MinHash实现,并且可能不适用于大规模数据集。在实际应用中,您可能需要使用更高效的数据结构和算法来处理大量数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值