1. 程式人生 > >simHash 簡介以及 java 實現

simHash 簡介以及 java 實現

傳統的 hash 演算法只負責將原始內容儘量均勻隨機地對映為一個簽名值,原理上相當於偽隨機數產生演算法。產生的兩個簽名,如果相等,說明原始內容在一定概 率 下是相等的;如果不相等,除了說明原始內容不相等外,不再提供任何資訊,因為即使原始內容只相差一個位元組,所產生的簽名也很可能差別極大。從這個意義 上來 說,要設計一個 hash 演算法,對相似的內容產生的簽名也相近,是更為艱難的任務,因為它的簽名值除了提供原始內容是否相等的資訊外,還能額外提供不相等的 原始內容的差異程度的資訊。
而 Google 的 simhash 演算法產生的簽名,可以滿足上述要求。出人意料,這個演算法並不深奧,其思想是非常清澈美妙的。

1、Simhash 演算法簡介

simhash演算法的輸入是一個向量,輸出是一個 f 位的簽名值。為了陳述方便,假設輸入的是一個文件的特徵集合,每個特徵有一定的權重。比如特徵可以是文件中的詞,其權重可以是這個詞出現的次數。 simhash 演算法如下:
1,將一個 f 維的向量 V 初始化為 0 ; f 位的二進位制數 S 初始化為 0 ;
2,對每一個特徵:用傳統的 hash 演算法對該特徵產生一個 f 位的簽名 b 。對 i=1 到 f :
如果b 的第 i 位為 1 ,則 V 的第 i 個元素加上該特徵的權重;
否則,V 的第 i 個元素減去該特徵的權重。 
3,如果 V 的第 i 個元素大於 0 ,則 S 的第 i 位為 1 ,否則為 0 ;

4,輸出 S 作為簽名。

2、演算法幾何意義和原理

這個演算法的幾何意義非常明瞭。它首先將每一個特徵對映為f維空間的一個向量,這個對映規則具體是怎樣並不重要,只要對很多不同的特徵來說,它們對所對應的向量是均勻隨機分佈的,並且對相同的特徵來說對應的向量是唯一的就行。比如一個特徵的4位hash簽名的二進位制表示為1010,那麼這個特徵對應的 4維向量就是(1, -1, 1, -1)T,即hash簽名的某一位為1,對映到的向量的對應位就為1,否則為-1。然後,將一個文件中所包含的各個特徵對應的向量加權求和,加權的係數等於該特徵的權重。得到的和向量即表徵了這個文件,我們可以用向量之間的夾角來衡量對應文件之間的相似度。最後,為了得到一個f位的簽名,需要進一步將其壓縮,如果和向量的某一維大於0,則最終簽名的對應位為1,否則為0。這樣的壓縮相當於只留下了和向量所在的象限這個資訊,而64位的簽名可以表示多達264個象限,因此只儲存所在象限的資訊也足夠表徵一個文件了。

明確了演算法了幾何意義,使這個演算法直觀上看來是合理的。但是,為何最終得到的簽名相近的程度,可以衡量原始文件的相似程度呢?這需要一個清晰的思路和證明。在simhash的發明人Charikar的論文中[2]並沒有給出具體的simhash演算法和證明,以下列出我自己得出的證明思路。

Simhash是由隨機超平面hash演算法演變而來的,隨機超平面hash演算法非常簡單,對於一個n維向量v,要得到一個f位的簽名(f<<n),演算法如下:
1,隨機產生f個n維的向量r1,…rf;
2,對每一個向量ri,如果v與ri的點積大於0,則最終簽名的第i位為1,否則為0.

這個演算法相當於隨機產生了f個n維超平面,每個超平面將向量v所在的空間一分為二,v在這個超平面上方則得到一個1,否則得到一個0,然後將得到的 f個0或1組合起來成為一個f維的簽名。如果兩個向量u, v的夾角為θ,則一個隨機超平面將它們分開的概率為θ/π,因此u, v的簽名的對應位不同的概率等於θ/π。所以,我們可以用兩個向量的簽名的不同的對應位的數量,即漢明距離,來衡量這兩個向量的差異程度。

Simhash演算法與隨機超平面hash是怎麼聯絡起來的呢?在simhash演算法中,並沒有直接產生用於分割空間的隨機向量,而是間接產生的:第 k個特徵的hash簽名的第i位拿出來,如果為0,則改為-1,如果為1則不變,作為第i個隨機向量的第k維。由於hash簽名是f位的,因此這樣能產生 f個隨機向量,對應f個隨機超平面。下面舉個例子:
假設用5個特徵w1,…,w5來表示所有文件,現要得到任意文件的一個3維簽名。假設這5個特徵對應的3維向量分別為:
h(w1) = (1, -1, 1)T
h(w2) = (-1, 1, 1)T
h(w3) = (1, -1, -1)T
h(w4) = (-1, -1, 1)T
h(w5) = (1, 1, -1)T

按simhash演算法,要得到一個文件向量d=(w1=1, w2=2, w3=0, w4=3, w5=0) T的簽名,

先要計算向量m = 1*h(w1) + 2*h(w2) + 0*h(w3) + 3*h(w4) + 0*h(w5) = (-4, -2, 6) T,
然後根據simhash演算法的步驟3,得到最終的簽名s=001。

上面的計算步驟其實相當於,先得到3個5維的向量,第1個向量由h(w1),…,h(w5)的第1維組成:

r1=(1,-1,1,-1,1) T;
第2個5維向量由h(w1),…,h(w5)的第2維組成:
r2=(-1,1,-1,-1,1) T;
同理,第3個5維向量為:
r3=(1,1,-1,1,-1) T.
按隨機超平面演算法的步驟2,分別求向量d與r1,r2,r3的點積:
d T r1=-4 < 0,所以s1=0;
d T r2=-2 < 0,所以s2=0;
d T r3=6 > 0,所以s3=1.
故最終的簽名s=001,與simhash演算法產生的結果是一致的。

從上面的計算過程可以看出,simhash演算法其實與隨機超平面hash演算法是相同的,simhash演算法得到的兩個簽名的漢明距離,可以用來衡量原始向量的夾角。這其實是一種降維技術,將高維的向量用較低維度的簽名來表徵。衡量兩個內容相似度,需要計算漢明距離,這對給定簽名查詢相似內容的應用來說帶來了一些計算上的困難;我想,是否存在更為理想的simhash演算法,原始內容的差異度,可以直接由簽名值的代數差來表示呢?

3、比較相似度

海明距離: 兩個碼字的對應位元取值不同的位元數稱為這兩個碼字的海明距離。一個有效編碼集中, 任意兩個碼字的海明距離的最小值稱為該編碼集的海明距離。舉例如下: 10101 和 00110 從第一位開始依次有第一位、第四、第五位不同,則海明距離為 3.

異或: 只有在兩個比較的位不同時其結果是1 ,否則結果為 0 

對每篇文件根據SimHash 算出簽名後,再計算兩個簽名的海明距離(兩個二進位制異或後 1 的個數)即可。根據經驗值,對 64 位的 SimHash ,海明距離在 3 以內的可以認為相似度比較高。
假設對64 位的 SimHash ,我們要找海明距離在 3 以內的所有簽名。我們可以把 64 位的二進位制簽名均分成 4塊,每塊 16 位。根據鴿巢原理(也成抽屜原理,見組合數學),如果兩個簽名的海明距離在 3 以內,它們必有一塊完全相同。
我們把上面分成的4 塊中的每一個塊分別作為前 16 位來進行查詢。 建立倒排索引。


如果庫中有2^34 個(大概 10 億)簽名,那麼匹配上每個塊的結果最多有 2^(34-16)=262144 個候選結果(假設資料是均勻分佈, 16 位的資料,產生的像限為 2^16 個,則平均每個像限分佈的文件數則 2^34/2^16 = 2^(34-16)),四個塊返回的總結果數為 4* 262144 (大概 100 萬)。原本需要比較 10 億次,經過索引,大概就只需要處理 100 萬次了。由此可見,確實大大減少了計算量。 

4、示例

程式碼1:

import java.io.IOException;
import java.io.StringReader;
import java.math.BigInteger;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;


import org.wltea.analyzer.core.IKSegmenter;
import org.wltea.analyzer.core.Lexeme;


/**
 * 
* @ProjectName gppos
* @ClassName SimHash 
* @Description TODO(判斷文字相似度,只支援中文) 
* @author makang
* @date 2016-5-27 下午4:57:56 
* @version V1.0
 */
public class SimHash {
private String tokens;
    private BigInteger intSimHash;
    private String strSimHash;
    private int hashbits = 64;
    public SimHash(String tokens) throws IOException {
        this.tokens = tokens;
        this.intSimHash = this.simHash();
    }
    public SimHash(String tokens, int hashbits) throws IOException {
        this.tokens = tokens;
        this.hashbits = hashbits;
        this.intSimHash = this.simHash();
    }
    HashMap<String, Integer> wordMap = new HashMap<String, Integer>();
    public BigInteger simHash() throws IOException {
        // 定義特徵向量/陣列
        int[] v = new int[this.hashbits];
        // 英文分詞
        // StringTokenizer stringTokens = new StringTokenizer(this.tokens);
        // while (stringTokens.hasMoreTokens()) {
        // String temp = stringTokens.nextToken();
        // }
        // 1、中文分詞,分詞器採用 IKAnalyzer3.2.8 ,僅供演示使用,新版 API 已變化。
        StringReader reader = new StringReader(this.tokens);
        // 當為true時,分詞器進行最大詞長切分
        IKSegmenter ik = new IKSegmenter(reader, true);
        Lexeme lexeme = null;
        String word = null;
       // String temp = null;
        while ((lexeme = ik.next()) != null) {
            word = lexeme.getLexemeText();
            // 注意停用詞會被幹掉
            // System.out.println(word);
            // 2、將每一個分詞hash為一組固定長度的數列.比如 64bit 的一個整數.
            BigInteger t = this.hash(word);
            for (int i = 0; i < this.hashbits; i++) {
                BigInteger bitmask = new BigInteger("1").shiftLeft(i);
                // 3、建立一個長度為64的整數陣列(假設要生成64位的數字指紋,也可以是其它數字),
                // 對每一個分詞hash後的數列進行判斷,如果是1000...1,那麼陣列的第一位和末尾一位加1,
                // 中間的62位減一,也就是說,逢1加1,逢0減1.一直到把所有的分詞hash數列全部判斷完畢.
                if (t.and(bitmask).signum() != 0) {
                    // 這裡是計算整個文件的所有特徵的向量和
                    // 這裡實際使用中需要 +- 權重,比如詞頻,而不是簡單的 +1/-1,
                    v[i] += 1;
                } else {
                    v[i] -= 1;
                }
            }
        }


        BigInteger fingerprint = new BigInteger("0");
        StringBuffer simHashBuffer = new StringBuffer();
        for (int i = 0; i < this.hashbits; i++) {
            // 4、最後對陣列進行判斷,大於0的記為1,小於等於0的記為0,得到一個 64bit 的數字指紋/簽名.
            if (v[i] >= 0) {
                fingerprint = fingerprint.add(new BigInteger("1").shiftLeft(i));
                simHashBuffer.append("1");
            } else {
                simHashBuffer.append("0");
            }
        }
        this.strSimHash = simHashBuffer.toString();
        System.out.println(this.strSimHash + " length " + this.strSimHash.length());
        return fingerprint;
    }


    private BigInteger hash(String source) {
        if (source == null || source.length() == 0) {
            return new BigInteger("0");
        } else {
            char[] sourceArray = source.toCharArray();
            BigInteger x = BigInteger.valueOf(((long) sourceArray[0]) << 7);
            BigInteger m = new BigInteger("1000003");
            BigInteger mask = new BigInteger("2").pow(this.hashbits).subtract(new BigInteger("1"));
            for (char item : sourceArray) {
                BigInteger temp = BigInteger.valueOf((long) item);
                x = x.multiply(m).xor(temp).and(mask);
            }
            x = x.xor(new BigInteger(String.valueOf(source.length())));
            if (x.equals(new BigInteger("-1"))) {
                x = new BigInteger("-2");
            }
            return x;
        }
    }


    public int hammingDistance(SimHash other) {


        BigInteger x = this.intSimHash.xor(other.intSimHash);
        int tot = 0;


        // 統計x中二進位制位數為1的個數
        // 我們想想,一個二進位制數減去1,那麼,從最後那個1(包括那個1)後面的數字全都反了,
        // 對吧,然後,n&(n-1)就相當於把後面的數字清0,
        // 我們看n能做多少次這樣的操作就OK了。


        while (x.signum() != 0) {
            tot += 1;
            x = x.and(x.subtract(new BigInteger("1")));
        }
        return tot;
    }


    public int getDistance(String str1, String str2) {
        int distance;
        if (str1.length() != str2.length()) {
            distance = -1;
        } else {
            distance = 0;
            for (int i = 0; i < str1.length(); i++) {
                if (str1.charAt(i) != str2.charAt(i)) {
                    distance++;
                }
            }
        }
        return distance;
    }


    public List<Object> subByDistance(SimHash simHash, int distance) {
        // 分成幾組來檢查
        int numEach = this.hashbits / (distance + 1);
        List<Object> characters = new ArrayList<Object>();


        StringBuffer buffer = new StringBuffer();


       // int k = 0;
        for (int i = 0; i < this.intSimHash.bitLength(); i++) {
            // 當且僅當設定了指定的位時,返回 true
            boolean sr = simHash.intSimHash.testBit(i);


            if (sr) {
                buffer.append("1");
            } else {
                buffer.append("0");
            }


            if ((i + 1) % numEach == 0) {
                // 將二進位制轉為BigInteger
                BigInteger eachValue = new BigInteger(buffer.toString(), 2);
                System.out.println("----" + eachValue);
                buffer.delete(0, buffer.length());
                characters.add(eachValue);
            }
        }


        return characters;
    }


    public static void main(String[] args) throws IOException {
        String s = "傳統的 hash 演算法只負責將原始內容儘量均勻隨機地對映為一個簽名值," 
                + "原理上相當於偽隨機數產生演算法。產生的兩個簽名,如果相等,說明原始內容在一定概 率 下是相等的;"
                + "如果不相等,除了說明原始內容不相等外,不再提供任何資訊,因為即使原始內容只相差一個位元組," 
                + "所產生的簽名也很可能差別極大。從這個意義 上來 說,要設計一個 hash 演算法,"
                + "對相似的內容產生的簽名也相近,是更為艱難的任務,因為它的簽名值除了提供原始內容是否相等的資訊外," 
                + "還能額外提供不相等的 原始內容的差異程度的資訊。";
        SimHash hash1 = new SimHash(s, 64);
        System.out.println(hash1.intSimHash + "  " + hash1.intSimHash.bitLength());
        // 計算 海明距離 在 3 以內的各塊簽名的 hash 值
        hash1.subByDistance(hash1, 3);


        // 刪除首句話,並加入兩個干擾串
        s = "原理上相當於偽隨機數產生演算法。產生的兩個簽名,如果相等,說明原始內容在一定概 率 下是相等的;"
                + "如果不相等,除了說明原始內容不相等外,不再提供任何資訊,因為即使原始內容只相差一個位元組," 
                + "所產生的簽名也很可能差別極大。從這個意義 上來 說,要設計一個 hash 演算法,"
                + "對相似的內容產生的簽名也相近,是更為艱難的任務,因為它的簽名值除了提供原始內容是否相等的資訊外," 
                + "干擾1還能額外提供不相等的 原始內容的差異程度的資訊。";
        SimHash hash2 = new SimHash(s, 64);
        System.out.println(hash2.intSimHash + "  " + hash2.intSimHash.bitCount());
        hash1.subByDistance(hash2, 3);


        // 首句前新增一句話,並加入四個干擾串
        s = "imhash演算法的輸入是一個向量,輸出是一個 f 位的簽名值。為了陳述方便," 
                + "假設輸入的是一個文件的特徵集合,每個特徵有一定的權重。"
                + "傳統干擾4的 hash 演算法只負責將原始內容儘量均勻隨機地對映為一個簽名值," 
                + "原理上這次差異有多大呢3相當於偽隨機數產生演算法。產生的兩個簽名,如果相等,"
                + "說明原始內容在一定概 率 下是相等的;如果不相等,除了說明原始內容不相等外,不再提供任何資訊,"
                + "因為即使原始內容只相差一個位元組,所產生的簽名也很可能差別極大。從這個意義 上來 說,"
                + "要設計一個 hash 演算法,對相似的內容產生的簽名也相近,是更為艱難的任務,因為它的簽名值除了提供原始"
                + "內容是否相等的資訊外,干擾1還能額外提供不相等的 原始再來干擾2內容的差異程度的資訊。";
        SimHash hash3 = new SimHash(s, 64);
        System.out.println(hash3.intSimHash + "  " + hash3.intSimHash.bitCount());
        hash1.subByDistance(hash3, 3);


        System.out.println("============================");


        int dis = hash1.getDistance(hash1.strSimHash, hash2.strSimHash);
        System.out.println(hash1.hammingDistance(hash2) + " " + dis);
        // 根據鴿巢原理(也成抽屜原理,見組合數學),如果兩個簽名的海明距離在 3 以內,它們必有一塊簽名subByDistance()完全相同。
        int dis2 = hash1.getDistance(hash1.strSimHash, hash3.strSimHash);
        System.out.println(hash1.hammingDistance(hash3) + " " + dis2);
    }
} 程式碼2:
/**
 * Function: 注:該示例程式暫不支援中文

 * Date:     2013-8-4 下午11:01:45 

 * @author   june: [email protected]
 */
import java.math.BigInteger;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.StringTokenizer;

public class SimHash {

	private String tokens;

	private BigInteger intSimHash;

	private String strSimHash;

	private int hashbits = 64;

	public SimHash(String tokens) {
		this.tokens = tokens;
		this.intSimHash = this.simHash();
	}

	public SimHash(String tokens, int hashbits) {
		this.tokens = tokens;
		this.hashbits = hashbits;
		this.intSimHash = this.simHash();
	}

	HashMap wordMap = new HashMap();

	public BigInteger simHash() {
		// 定義特徵向量/陣列
		int[] v = new int[this.hashbits];
		// 1、將文字去掉格式後, 分詞.
		StringTokenizer stringTokens = new StringTokenizer(this.tokens);
		while (stringTokens.hasMoreTokens()) {
			String temp = stringTokens.nextToken();
			// 2、將每一個分詞hash為一組固定長度的數列.比如 64bit 的一個整數.
			BigInteger t = this.hash(temp);
			for (int i = 0; i < this.hashbits; i++) {
				BigInteger bitmask = new BigInteger("1").shiftLeft(i);
				// 3、建立一個長度為64的整數陣列(假設要生成64位的數字指紋,也可以是其它數字),
				// 對每一個分詞hash後的數列進行判斷,如果是1000...1,那麼陣列的第一位和末尾一位加1,
				// 中間的62位減一,也就是說,逢1加1,逢0減1.一直到把所有的分詞hash數列全部判斷完畢.
				if (t.and(bitmask).signum() != 0) {
					// 這裡是計算整個文件的所有特徵的向量和
					// 這裡實際使用中需要 +- 權重,而不是簡單的 +1/-1,
					v[i] += 1;
				} else {
					v[i] -= 1;
				}
			}
		}
		BigInteger fingerprint = new BigInteger("0");
		StringBuffer simHashBuffer = new StringBuffer();
		for (int i = 0; i < this.hashbits; i++) {
			// 4、最後對陣列進行判斷,大於0的記為1,小於等於0的記為0,得到一個 64bit 的數字指紋/簽名.
			if (v[i] >= 0) {
				fingerprint = fingerprint.add(new BigInteger("1").shiftLeft(i));
				simHashBuffer.append("1");
			} else {
				simHashBuffer.append("0");
			}
		}
		this.strSimHash = simHashBuffer.toString();
		System.out.println(this.strSimHash + " length " + this.strSimHash.length());
		return fingerprint;
	}

	private BigInteger hash(String source) {
		if (source == null || source.length() == 0) {
			return new BigInteger("0");
		} else {
			char[] sourceArray = source.toCharArray();
			BigInteger x = BigInteger.valueOf(((long) sourceArray[0]) << 7);
			BigInteger m = new BigInteger("1000003");
			BigInteger mask = new BigInteger("2").pow(this.hashbits).subtract(new BigInteger("1"));
			for (char item : sourceArray) {
				BigInteger temp = BigInteger.valueOf((long) item);
				x = x.multiply(m).xor(temp).and(mask);
			}
			x = x.xor(new BigInteger(String.valueOf(source.length())));
			if (x.equals(new BigInteger("-1"))) {
				x = new BigInteger("-2");
			}
			return x;
		}
	}

	public int hammingDistance(SimHash other) {

		BigInteger x = this.intSimHash.xor(other.intSimHash);
		int tot = 0;

		// 統計x中二進位制位數為1的個數
		// 我們想想,一個二進位制數減去1,那麼,從最後那個1(包括那個1)後面的數字全都反了,對吧,然後,n&(n-1)就相當於把後面的數字清0,
		// 我們看n能做多少次這樣的操作就OK了。

		while (x.signum() != 0) {
			tot += 1;
			x = x.and(x.subtract(new BigInteger("1")));
		}
		return tot;
	}

	public int getDistance(String str1, String str2) {
		int distance;
		if (str1.length() != str2.length()) {
			distance = -1;
		} else {
			distance = 0;
			for (int i = 0; i < str1.length(); i++) {
				if (str1.charAt(i) != str2.charAt(i)) {
					distance++;
				}
			}
		}
		return distance;
	}

	public List subByDistance(SimHash simHash, int distance) {
		// 分成幾組來檢查
		int numEach = this.hashbits / (distance + 1);
		List characters = new ArrayList();

		StringBuffer buffer = new StringBuffer();

		int k = 0;
		for (int i = 0; i < this.intSimHash.bitLength(); i++) {
			// 當且僅當設定了指定的位時,返回 true
			boolean sr = simHash.intSimHash.testBit(i);

			if (sr) {
				buffer.append("1");
			} else {
				buffer.append("0");
			}

			if ((i + 1) % numEach == 0) {
				// 將二進位制轉為BigInteger
				BigInteger eachValue = new BigInteger(buffer.toString(), 2);
				System.out.println("----" + eachValue);
				buffer.delete(0, buffer.length());
				characters.add(eachValue);
			}
		}

		return characters;
	}

	public static void main(String[] args) {
		String s = "This is a test string for testing";
		SimHash hash1 = new SimHash(s, 64);
		System.out.println(hash1.intSimHash + "  " + hash1.intSimHash.bitLength());
		hash1.subByDistance(hash1, 3);

		s = "This is a test string for testing, This is a test string for testing abcdef";
		SimHash hash2 = new SimHash(s, 64);
		System.out.println(hash2.intSimHash + "  " + hash2.intSimHash.bitCount());
		hash1.subByDistance(hash2, 3);
		
		s = "This is a test string for testing als";
		SimHash hash3 = new SimHash(s, 64);
		System.out.println(hash3.intSimHash + "  " + hash3.intSimHash.bitCount());
		hash1.subByDistance(hash3, 4);
		
		System.out.println("============================");
		
		int dis = hash1.getDistance(hash1.strSimHash, hash2.strSimHash);
		System.out.println(hash1.hammingDistance(hash2) + " " + dis);

		int dis2 = hash1.getDistance(hash1.strSimHash, hash3.strSimHash);
		System.out.println(hash1.hammingDistance(hash3) + " " + dis2);
		
		//通過Unicode編碼來判斷中文
		/*String str = "中國chinese";
		for (int i = 0; i < str.length(); i++) {
			System.out.println(str.substring(i, i + 1).matches("[\\u4e00-\\u9fbb]+"));
		}*/

	}
}

5、適用場景:

simHash在短文字的可行性:

測試相似文字的相似度與漢明距離
測試文字:20個城市名作為詞串:北京,上海,香港,深圳,廣州,臺北,南京,大連,蘇州,青島,無錫,佛山,重慶,寧波,杭州,成都,武漢,澳門,天津,瀋陽

相似度矩陣:

simHash碼:

勘誤:0.667, Hm:13 是對比的msg 1與2。
可見:相似度在0.8左右的Hamming距離為7,只有相似度高到0.9412,Hamming距離才近到4,

此時,反觀Google對此演算法的應用場景:網頁近重複、映象網站、內容複製、嵌入廣告、計數改變、少量修改。
以上原因對於長文字來說造成的相似度都會比較高,而對於短文字來說,如何處理海量資料的相似度文字更為合適的?

測試短文字(長度在8箇中文字元~45箇中文字元之間)相似性的誤判率如下圖所示:


REF:

1、simHash 簡介以及java實現

2、對simhash演算法的一些思考

3、Simhash演算法原理和網頁查重應用

4、其它

求二進位制數中1的個數

海量資料相似度計算之simhash短文字查詢(提升查詢效率)


相關推薦

simHash 簡介以及 java 實現

傳統的 hash 演算法只負責將原始內容儘量均勻隨機地對映為一個簽名值,原理上相當於偽隨機數產生演算法。產生的兩個簽名,如果相等,說明原始內容在一定概 率 下是相等的;如果不相等,除了說明原始內容不相等外,不再提供任何資訊,因為即使原始內容只相差一個位元組,所產生的簽名也很可能差別極大。從這個意義 上來

約瑟夫環簡介,問題以及java實現

問題:一群猴子排成一圈,按1,2,…….,n依次編號。然後從第一隻開始數,數到第m只,把它踢出圈,從它後面再開始數,再數到第m只,再把它踢出去………………….,如此不停的進行下去,直到最後只剩下一隻猴子為止,那隻猴子就叫做大王。要求:輸入m,n,輸出最後的那個大

八大排序算法原理以及Java實現(直接插入排序)

不能 oat 設立 side 堆排 八大排序 算法 line load 概述 排序有內部排序和外部排序,內部排序是數據記錄在內存中進行排序,而外部排序是因排序的數據很大,一次不能容納全部的排序記錄,在排序過程中需要訪問外存。 我們這裏說說八大排序就是內部排序。

大整數相乘問題總結以及Java實現

語言 java 變換 hal () 以及 pow divide 優化 最近在跟coursera上斯坦福大學的算法專項課,其中開篇提到了兩個整數相乘的問題,其中最簡單的方法就是模擬我們小學的整數乘法,可想而知這不是比較好的算法,這門課可以說非常棒,帶領我們不斷探索更優的算法,

Hash表分析以及Java實現

   這篇部落格主要探討Hash表中的一些原理/概念,及根據這些原理/概念,自己設計一個用來存放/查詢資料的Hash表,並且與JDK中的HashMap類進行比較。 我們分一下七個步驟來進行。  一。    Hash表概念 二 . &n

BFS和DFS詳解以及java實現(轉載)

作者: Leo-Yang 原文都先發布在作者個人部落格: http://www.leoyang.net/ 本文版權歸作者和部落格園共有,歡迎轉載,但未經作者同意必須保留此段宣告,且在文章頁面明顯位置給出原文連線,否則保留追究法律責任的權利. 前言

深入剖析紅黑樹以及JAVA實現

作者:美團技術團隊 連結:https://zhuanlan.zhihu.com/p/24367771 來源:知乎 著作權歸作者所有。商業轉載請聯絡作者獲得授權,非商業轉載請註明出處。   紅黑樹是平衡二叉查詢樹的一種。為了深入理解紅黑樹,我們需要從二叉查詢樹開始講起。 BST

MapReduce:原理之Word Count 以及Java實現

為什麼要設定記憶體緩衝區?    批量收集map的結果,減少磁碟IO次數,提高效率。 磁碟檔案要寫到哪裡?     寫磁碟將按照輪詢方式寫到mapred.local.dir屬性指定的作業特定子目錄的目錄中。也就是存放在TaskTracker夠得著的某個本地目錄,每一個

AVL樹簡介Java實現

AVL樹簡介 AVL樹是被最先發明的一種較為簡單的平衡二叉查詢樹。 它的特點是: 1.本身首先是一棵二叉查詢樹。 2.帶有平衡條件:每個結點的左右子樹的高度之差的絕對值(平衡因子LoadFactor)最多為1。(設根結點的高度為1) 上面的兩張圖片,左邊的是AV

各路排序神仙理解以及Java實現

啊啊啊啊啊啊!大話資料結構終於看到了排序這一章章啦,然而各路排序神仙,真的是看懂了!僅僅是看懂了,所以打算用兩天的時間比幾個常見的演算法寫一遍啦,加深一下印象。嚶嚶嚶 氣泡排序 這個演算法我記得是上C語言就接觸了,他的思想是,遇到逆序對,就進行交換啊,就這樣慢慢的第n個有序,第n-1個有序......

redis單機版的安裝部署以及java實現

2018年11月05日 20:39:57 孤獨求劍 閱讀數:1 個人分類: dedis

紅黑樹原理解析以及Java實現

紅黑樹 本文的主要內容: 1、紅黑樹的基本概念以及最重要的5點規則。 2、紅黑樹的左旋轉、右旋轉、重新著色的原理與Java實現; 3、紅黑樹的增加結點、刪除結點過程解析; 1.紅黑樹的基本概念與資料結構表示 首先紅黑樹來個定義: 紅黑樹定

基數排序詳解以及java實現

前言 基數排序(radix sort)又稱桶排序(bucket sort),相對於常見的比較排序,基數排序是一種分配式排序,即通過將所有數字分配到應在的位置最後再覆蓋到原陣列完成排序的過程。我在上一篇講到的計數排序也屬於這種排序模式,上一篇結尾處提到了計數排序的穩定性,即排序前和排序後相同的數字相對位置保持

Dijkstra演算法以及java實現_02(程式碼部分)

1.資料庫表的設計: dijsname1和dijsname2是有向圖中的兩個點。 qinmid是這兩個點之間的親密度,將親密度的值作為邊的權值大小。 2.對資料庫中資料進行增刪改查操作: //這裡主要實現了insert(插入)操作和查詢操作(通

排序演算法之氣泡排序的思想以及Java實現

1 基本思想 設排序表長為n,從後向前或者從前向後兩兩比較相鄰元素的值,如果兩者的相對次序不對(A[i-1] > A[i]),則交換它們,其結果是將最小的元素交換到待排序序列的第一個位置,我們

leveldb簡介java實現demo

簡介 1.簡介 Leveldb是一個google實現的非常高效的kv資料庫,目前的版本1.2能夠支援billion級別的資料量了。 在這個數量級別下還有著非常高的效能,主要歸功於它的良好的設計。特別是LSM演算法。 2特點 LevelDB 是單程序的服務,效能非常之高

常見的簡單負載均衡演算法以及Java實現

讀完本文你將知道:  1. 什麼是負載均衡?  2. 負載均衡的幾種簡單實現:  (1) 輪詢法(Round Robin)  (2)隨機法(Random)  (3)源地址Hash法(Hash)  (4)加權輪詢法(Weight Round Robin)  (5)加權隨機

排序演算法之直接插入排序的思想以及Java實現

1,基本思想 假設待排序的資料是陣列A[1….n]。初始時,A[1]自成1個有序區,無序區為A[2….n]。在排序的過程中,依次將A[i] (i=2,3,….,n)從後往前插入到前面已排好序的子陣列A

BFS和DFS詳解以及java實現

前言 圖在演算法世界中的重要地位是不言而喻的,曾經看到一篇Google的工程師寫的一篇《Get that job at Google!》文章中說到面試官問的問題中幾乎有一半的問題都可以用圖的方法去解決。由此也可以看出圖確實適用範圍確實很廣。 圖的表示 閒話不多說,首先要

機器學習演算法——PCA演算法介紹以及Java實現

PCA演算法 一、演算法概述 主成分分析(PCA)是多元統計分析中用來分析資料的一種方法,PCA通過線性變換將原始資料變換為一組各維度線性無關的表示,可用於提取資料的主要特徵分量,常用於高維資料的降維。 PCA方法最著名的應用應該是在人臉識別中特徵提取及資