当前位置:   article > 正文

布隆过滤器详解及java代码实现_java实现布隆过滤器

java实现布隆过滤器

1.简介

布隆过滤器(Bloom Filter)是1970年由布隆提出的。它实际上是一个很长的二进制向量和一系列随机映射函数。布隆过滤器可以用于检索一个元素是否在一个集合中。它的优点是空间效率和查询时间都比一般的算法要好的多,缺点是有一定的误识别率和删除困难。

2.基本思想

如果想要判断一个元素是不是在一个集合里,一般想到的是将所有元素保存起来,然后通过比较确定。链表,树等等数据结构都是这种思路. 但是随着集合中元素的增加,我们需要的存储空间越来越大,检索速度也越来越慢(O(n),O(logn))。不过世界上还有一种叫作散列表(又叫哈希表,Hash table)的数据结构。它可以通过一个Hash函数将一个元素映射成一个位阵列(Bit array)中的一个点。这样一来,我们只要看看这个点是不是1就可以知道集合中有没有它了。这就是布隆过滤器的基本思想。

3.特性

  • 如果根据同一个哈希函数得到的哈希值不同,那么这两个哈希值的原始输入值肯定不同。
  • 如果根据同一个哈希函数得到的两个哈希值相等,两个哈希值的原始输入值有可能相等,有可能不相等。

这就类似于 Java 中两个对象的 HashCode 相等,但是对象本身不一定相等的道理。说白了,通过散列函数计算后得到位数组上映射点的值全都是1,不一定是要查询的这个变量之前存进来时设置的,也有可能是其他元素映射的点。这也就引出了布隆过滤器的一个特性:存在一定的误判

4.总结

优点:在空间和时间方面,都有着巨大的优势。因为不是存完整的数据,是一个二进制向量,能节省大量的内存空间,时间复杂度方面,由于计算时是根据散列函数计算查询的,那么假设有N个散例函数,那么时间复杂度就是O(N);同时在存储元素时存储的不是元素本身,而是二进制向量,所以在一些对保密性要求严格的场景有一定优势。

缺点:存在一定的误判(存进布隆过滤器里的元素越多,误判率越高);不能删除布隆过滤器里的元素,随着使用的时间越来越长,因为不能删除,存进里面的元素越来越多,导致占用内存越来越多,误判率越来越高,最后不得不重置。

5.应用

  • 网页爬虫对URL的去重,避免爬去相同的URL地址。
  • 垃圾邮件过滤,从数十亿个垃圾邮件列表中判断某邮箱是否是杀垃圾邮箱。
  • 解决数据库缓存击穿,黑客攻击服务器时,会构建大量不存在于缓存中的key向服务器发起请求,在数据量足够大的时候,频繁的数据库查询会导致挂机。
  • 秒杀系统,查看用户是否重复购买。

6.代码实现

  1. import java.util.BitSet;
  2. public class MyBloomFilter {
  3. /**
  4. * 一个长度为10亿的比特位
  5. */
  6. private static final int DEFAULT_SIZE = 256 << 22;
  7. private static final int[] seeds = {3, 5, 7, 11, 13, 31, 37, 61};
  8. private static HashFunction[] functions = new HashFunction[seeds.length];
  9. private static BitSet bitset = new BitSet(DEFAULT_SIZE);
  10. static {
  11. for (int i = 0; i < functions.length; i++) {
  12. functions[i]=new HashFunction(DEFAULT_SIZE,seeds[i]);
  13. }
  14. }
  15. public static void add(String value) {
  16. if (value != null) {
  17. for (HashFunction f : functions) {
  18. //计算 hash 值并修改 bitmap 中相应位置为 true
  19. bitset.set(f.hash(value), true);
  20. }
  21. }
  22. }
  23. public static boolean contains(String value){
  24. if(value==null){
  25. return false;
  26. }
  27. boolean ret=true;
  28. for (HashFunction f : functions) {
  29. ret=bitset.get(f.hash(value));
  30. if(!ret){
  31. break;
  32. }
  33. }
  34. return ret;
  35. }
  36. /**
  37. * 测试
  38. * @param args
  39. */
  40. public static void main(String[] args) {
  41. for (int i = 0; i < 100000000; i++){
  42. add(i+"");
  43. }
  44. String id1=1234567+"";
  45. String id2=123456789+"";
  46. System.out.println(id1+" "+contains(id1));
  47. System.out.println(id2+" "+contains(id2));
  48. }
  49. static class HashFunction{
  50. private int size;
  51. private int seed;
  52. public HashFunction(int size, int seed) {
  53. this.size = size;
  54. this.seed = seed;
  55. }
  56. public int hash(String value){
  57. int result=0;
  58. int len=value.length();
  59. for (int i = 0; i < len; i++) {
  60. result=seed*result+value.charAt(i);
  61. }
  62. return (size-1)&result;
  63. }
  64. }
  65. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小蓝xlanll/article/detail/382672?site
推荐阅读
相关标签
  

闽ICP备14008679号