你們好,我是jack xu,不知道從何時開始,原本默默無聞的布隆過濾器一會兒名聲大燥,在面試中面試官問到怎麼避免緩存穿透,你的第一反應可能就是布隆過濾器,緩存穿透=布隆過濾器成了標配,但具體什麼是布隆過濾器,怎麼使用布隆過濾器不是很清楚,那今天咱們就來把他說清楚,講明白。。java
你們看下這幅圖,用戶可能進行了一次條件錯誤的查詢,這時候redis是不存在的,按照常規流程就是去數據庫找了,但是這是一次錯誤的條件查詢,數據庫固然也不會存在,也不會往redis裏面寫值,返回給用戶一個空,這樣的操做一次兩次還好,但是次數多了還了得,我放redis原本就是爲了擋一擋,減輕數據庫的壓力,如今redis變成了形同虛設,每次仍是去數據庫查找了,這個就叫作緩存穿透,至關於redis不存在了,被擊穿了,對於這種狀況很好解決,咱們能夠在redis緩存一個空字符串或者特殊字符串,好比&&,下次咱們去redis中查詢的時候,當取到的值是空或者&&,咱們就知道這個值在數據庫中是沒有的,就不會在去數據庫中查詢,ps:這裏緩存不存在key的時候必定要設置過時時間,否則當數據庫已經新增了這一條記錄的時候,這樣會致使緩存和數據庫不一致的狀況web
上面這個是重複查詢同一個不存在的值的狀況,若是應用每次查詢的不存在的值是不同的呢?即便你每次都緩存特殊字符串也沒用,由於它的值不同,好比咱們的數據庫用戶id是111,112,113,114依次遞增,可是別人要攻擊你,故意拿-100,-936,-545這種亂七八糟的key來查詢,這時候redis和數據庫這種值都是不存在的,人家每次拿的key也不同,你就算緩存了也沒用,這時候數據庫的壓力是至關大,比上面這種狀況可怕的多,怎麼辦呢,這時候咱們今天的主角布隆過濾器就登場了。。面試
問:如何在海量元素中(例如 10 億無序、不定長、不重複)快速判斷一個元素是否存在?好,咱們最簡單的想法就是把這麼多數據放到數據結構裏去,好比List、Map、Tree,一搜不就出來了嗎,好比map.get(),咱們假設一個元素1個字節的字段,10億的數據大概須要 900G 的內存空間,這個對於普通的服務器來講是承受不了的,固然面試官也不但願聽到你這個答案,由於太笨了吧,咱們確定是要用一種好的方法,巧妙的方法來解決,這裏引入一種節省空間的數據結構,位圖,他是一個有序的數組,只有兩個值,0 和 1。0表明不存在,1表明存在。redis
有了這個屌炸天的東西,如今咱們還須要一個映射關係,你總得知道某個元素在哪一個位置上吧,而後在去看這個位置上是0仍是1,怎麼解決這個問題呢,那就要用到哈希函數,用哈希函數有兩個好處,第一是哈希函數不管輸入值的長度是多少,獲得的輸出值長度是固定的,第二是他的分佈是均勻的,若是全擠的一塊去那還怎麼區分,好比MD五、SHA-1這些就是常見的哈希算法。算法
咱們經過哈希函數計算之後就能夠到相應的位置去找是否存在了,咱們看紅色的線,24和147通過哈希函數獲得的哈希值是同樣的,咱們把這種狀況叫作哈希衝突或者哈希碰撞。哈希碰撞是不可避免的,咱們能作的就是下降哈希碰撞的機率,第一種是能夠擴大維數組的長度或者說位圖容量,由於咱們的函數是分佈均勻的,因此位圖容量越大,在同一個位置發生哈希碰撞的機率就越小。可是越大的位圖容量,意味着越多的內存消耗,因此咱們想一想能不能經過其餘的方式來解決,第二種方式就是通過多幾個哈希函數的計算,你想啊,24和147如今通過一次計算就碰撞了,那我通過5次,10次,100次計算還能碰撞的話那真的是緣分了,大家能夠在一塊兒了,但也不是越屢次哈希函數計算越好,由於這樣很快就會填滿位圖,並且計算也是須要消耗時間,因此咱們須要在時間和空間上尋求一個平衡。。數據庫
固然,這個事情早就有人研究過了,在 1970 年的時候,有一個叫作布隆的前輩對於判斷海量元素中元素是否存在的問題進行了研究,也就是到底須要多大的位圖容量和多少個哈希函數,它發表了一篇論文,提出的這個容器就叫作布隆過濾器。json
你們來看下這個圖,咱們看集合裏面3個元素,如今咱們要存了,好比說a,通過f1(a),f2(a),f3(a)通過三個哈希函數的計算,在相應的位置上存入1,元素b,c也是經過這三個函數計算放入相應的位置。當取的時候,元素a經過f1(a)函數計算,發現這個位置上是1,沒問題,第二個位置也是1,第三個位置上也是 1,這時候咱們說這個a在布隆過濾器中是存在的,沒毛病,同理咱們看下面的這個d,經過三次計算髮現獲得的結果也都是1,那麼咱們能說d在布隆過濾器中是存在的嗎,顯然是不行的,咱們仔細看d獲得的三個1實際上是f1(a),f1(b),f2(c)存進去的,並非d本身存進去的,這個仍是哈希碰撞致使的,咱們把這種原本不存在布隆過濾器中的元素誤判爲存在的狀況叫作假陽性(False Positive Probability,FPP)。網頁爬蟲
咱們再來看另外一個元素,e 元素。咱們要判斷它在容器裏面是否存在,同樣地要用這三個函數去計算。第一個位置是 1,第二個位置是 1,第三個位置是 0。那麼e元素能不能判斷是否在布隆過濾器中? 答案是確定的,e必定不存在。你想啊,若是e存在的話,他存進去的時候這三個位置都置爲1,如今查出來有一個位置是0,證實他沒存進去啊。。經過上面這張圖加說明,咱們得出兩個重要的結論數組
從容器的角度來講:緩存
若是布隆過濾器判斷元素在集合中存在,不必定存在 若是布隆過濾器判斷不存在,必定不存在
從元素的角度來講:
若是元素實際存在,布隆過濾器必定判斷存在 若是元素實際不存在,布隆過濾器可能判斷存在
小夥們請牢記
java爲何寫的人多,基數大,由於是開源的,擁抱開源,框架多,輪子多,並且一個功能的輪子還不止一個,光序列化就有fastjson,jackson,gson,隨你挑任你選,那布隆過濾器的輪子就是google提供的guava,咱們用代碼來看一下使用方法
首先引入咱們的架包
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>21.0</version>
</dependency>
複製代碼
這裏先往布隆過濾器裏面存放100萬個元素,而後分別測試100個存在的元素和9900個不存在的元素他們的正確率和誤判率
//插入多少數據
private static final int insertions = 1000000;
//指望的誤判率
private static double fpp = 0.02;
public static void main(String[] args) {
//初始化一個存儲string數據的布隆過濾器,默認誤判率是0.03
BloomFilter<String> bf = BloomFilter.create(Funnels.stringFunnel(Charsets.UTF_8), insertions, fpp);
//用於存放全部實際存在的key,用因而否存在
Set<String> sets = new HashSet<String>(insertions);
//用於存放全部實際存在的key,用於取出
List<String> lists = new ArrayList<String>(insertions);
//插入隨機字符串
for (int i = 0; i < insertions; i++) {
String uuid = UUID.randomUUID().toString();
bf.put(uuid);
sets.add(uuid);
lists.add(uuid);
}
int rightNum = 0;
int wrongNum = 0;
for (int i = 0; i < 10000; i++) {
// 0-10000之間,能夠被100整除的數有100個(100的倍數)
String data = i % 100 == 0 ? lists.get(i / 100) : UUID.randomUUID().toString();
//這裏用了might,看上去不是很自信,因此若是布隆過濾器判斷存在了,咱們還要去sets中實錘
if (bf.mightContain(data)) {
if (sets.contains(data)) {
rightNum++;
continue;
}
wrongNum++;
}
}
BigDecimal percent = new BigDecimal(wrongNum).divide(new BigDecimal(9900), 2, RoundingMode.HALF_UP);
BigDecimal bingo = new BigDecimal(9900 - wrongNum).divide(new BigDecimal(9900), 2, RoundingMode.HALF_UP);
System.out.println("在100W個元素中,判斷100個實際存在的元素,布隆過濾器認爲存在的:" + rightNum);
System.out.println("在100W個元素中,判斷9900個實際不存在的元素,誤認爲存在的:" + wrongNum + ",命中率:" + bingo + ",誤判率:" + percent);
}
複製代碼
最後得出的結果
咱們看到這個結果正是印證了上面的結論,這100個真實存在元素在布隆過濾器中必定存在,另外9900個不存在的元素,布隆過濾器仍是判斷了216個存在,這個就是誤判,緣由上面也說過了,因此布隆過濾器不是萬能的,可是他能幫咱們抵擋掉大部分不存在的數據已經很不錯了,已經減輕數據庫不少壓力了,另外誤判率0.02是在初始化布隆過濾器的時候咱們本身設的,若是不設默認是0.03,咱們本身設的時候千萬不能設0!
上面使用guava實現布隆過濾器是把數據放在本地內存中,咱們項目每每是分佈式的,咱們還能夠把數據放在redis中,用redis來實現布隆過濾器,這就須要咱們本身設計映射函數,本身度量二進制向量的長度,下面貼代碼,你們能夠直接拿來用的,已經通過測試了。。
/**
* 布隆過濾器核心類
*
* @param <T>
* @author jack xu
*/
public class BloomFilterHelper<T> {
private int numHashFunctions;
private int bitSize;
private Funnel<T> funnel;
public BloomFilterHelper(int expectedInsertions) {
this.funnel = (Funnel<T>) Funnels.stringFunnel(Charset.defaultCharset());
bitSize = optimalNumOfBits(expectedInsertions, 0.03);
numHashFunctions = optimalNumOfHashFunctions(expectedInsertions, bitSize);
}
public BloomFilterHelper(Funnel<T> funnel, int expectedInsertions, double fpp) {
this.funnel = funnel;
bitSize = optimalNumOfBits(expectedInsertions, fpp);
numHashFunctions = optimalNumOfHashFunctions(expectedInsertions, bitSize);
}
public int[] murmurHashOffset(T value) {
int[] offset = new int[numHashFunctions];
long hash64 = Hashing.murmur3_128().hashObject(value, funnel).asLong();
int hash1 = (int) hash64;
int hash2 = (int) (hash64 >>> 32);
for (int i = 1; i <= numHashFunctions; i++) {
int nextHash = hash1 + i * hash2;
if (nextHash < 0) {
nextHash = ~nextHash;
}
offset[i - 1] = nextHash % bitSize;
}
return offset;
}
/**
* 計算bit數組長度
*/
private int optimalNumOfBits(long n, double p) {
if (p == 0) {
p = Double.MIN_VALUE;
}
return (int) (-n * Math.log(p) / (Math.log(2) * Math.log(2)));
}
/**
* 計算hash方法執行次數
*/
private int optimalNumOfHashFunctions(long n, long m) {
return Math.max(1, (int) Math.round((double) m / n * Math.log(2)));
}
}
複製代碼
這裏在操做redis的位圖bitmap,你可能只知道redis五種數據類型,string,list,hash,set,zset,沒聽過bitmap,可是沒關係,你能夠說他是一種新的數據類型,也能夠說不是,由於他的本質仍是string,後面我也會專門寫一篇文章來介紹數據類型以及在他們在互聯網中的使用場景。。
/**
* redis操做布隆過濾器
*
* @param <T>
* @author xhj
*/
public class RedisBloomFilter<T> {
@Autowired
private RedisTemplate redisTemplate;
/**
* 刪除緩存的KEY
*
* @param key KEY
*/
public void delete(String key) {
redisTemplate.delete(key);
}
/**
* 根據給定的布隆過濾器添加值,在添加一個元素的時候使用,批量添加的性能差
*
* @param bloomFilterHelper 布隆過濾器對象
* @param key KEY
* @param value 值
* @param <T> 泛型,能夠傳入任何類型的value
*/
public <T> void add(BloomFilterHelper<T> bloomFilterHelper, String key, T value) {
int[] offset = bloomFilterHelper.murmurHashOffset(value);
for (int i : offset) {
redisTemplate.opsForValue().setBit(key, i, true);
}
}
/**
* 根據給定的布隆過濾器添加值,在添加一批元素的時候使用,批量添加的性能好,使用pipeline方式(若是是集羣下,請使用優化後RedisPipeline的操做)
*
* @param bloomFilterHelper 布隆過濾器對象
* @param key KEY
* @param valueList 值,列表
* @param <T> 泛型,能夠傳入任何類型的value
*/
public <T> void addList(BloomFilterHelper<T> bloomFilterHelper, String key, List<T> valueList) {
redisTemplate.executePipelined(new RedisCallback<Long>() {
@Override
public Long doInRedis(RedisConnection connection) throws DataAccessException {
connection.openPipeline();
for (T value : valueList) {
int[] offset = bloomFilterHelper.murmurHashOffset(value);
for (int i : offset) {
connection.setBit(key.getBytes(), i, true);
}
}
return null;
}
});
}
/**
* 根據給定的布隆過濾器判斷值是否存在
*
* @param bloomFilterHelper 布隆過濾器對象
* @param key KEY
* @param value 值
* @param <T> 泛型,能夠傳入任何類型的value
* @return 是否存在
*/
public <T> boolean contains(BloomFilterHelper<T> bloomFilterHelper, String key, T value) {
int[] offset = bloomFilterHelper.murmurHashOffset(value);
for (int i : offset) {
if (!redisTemplate.opsForValue().getBit(key, i)) {
return false;
}
}
return true;
}
}
複製代碼
最後就是測試類了
public static void main(String[] args) {
RedisBloomFilter redisBloomFilter = new RedisBloomFilter();
int expectedInsertions = 1000;
double fpp = 0.1;
redisBloomFilter.delete("bloom");
BloomFilterHelper<CharSequence> bloomFilterHelper = new BloomFilterHelper<>(Funnels.stringFunnel(Charset.defaultCharset()), expectedInsertions, fpp);
int j = 0;
// 添加100個元素
List<String> valueList = new ArrayList<>();
for (int i = 0; i < 100; i++) {
valueList.add(i + "");
}
long beginTime = System.currentTimeMillis();
redisBloomFilter.addList(bloomFilterHelper, "bloom", valueList);
long costMs = System.currentTimeMillis() - beginTime;
log.info("布隆過濾器添加{}個值,耗時:{}ms", 100, costMs);
for (int i = 0; i < 1000; i++) {
boolean result = redisBloomFilter.contains(bloomFilterHelper, "bloom", i + "");
if (!result) {
j++;
}
}
log.info("漏掉了{}個,驗證結果耗時:{}ms", j, System.currentTimeMillis() - beginTime);
}
複製代碼
注意這裏用的是addList,他的底層是pipelining管道,而add方法的底層是一個個for循環的setBit,這樣的速度效率是很慢的,可是他能有返回值,知道是否插入成功,而pipelining是不知道的,因此具體選擇用哪種方法看你的業務場景,以及須要插入的速度決定。。
第一步是將數據庫全部的數據加載到布隆過濾器。第二步當有請求來的時候先去布隆過濾器查詢,若是bf說沒有,第三步直接返回。若是bf說有,在往下走以前的流程。ps:另外guava的數據加載中只有put方法,小夥們能夠想下布隆過濾器中數據刪除和修改怎麼辦,爲何沒有delete的方法?
好,布隆過濾器到這裏就結束了,之後在面試中面試官在問到緩存擊穿怎麼辦,我相信你應該可以回答的頭頭是道了,就像我這樣通俗易懂的說出來便可,而後在工做中也能夠應用,好比鑑權服務,當用戶登陸的時候能夠先用布隆過濾器判斷下,而不是直接去redis、數據庫查,最後原創不易,若是你以爲寫的不錯,請點個贊哦。。