布隆过滤器

系统管理员 2022-03-25 07:37 411阅读 0赞

布隆过滤器原理

布隆过滤器有什么用?

布隆过滤器是可以用于判断一个元素是不是在一个集合里,并且相比于其它的数据结构,布隆过滤器在空间和时间方面都有巨大的优势。

特点:

  • 巴顿.布隆于一九七零年提出
  • 一个很长的二进制向量 (位数组)
  • 一系列随机函数 (哈希)
  • 空间效率和查询效率高:O(1)
  • 有一定的误判率(哈希表是精确匹配)

实现原理

布隆过滤器(Bloom Filter)的核心实现是一个超大的位数组和几个哈希函数。假设位数组的长度为m,哈希函数的个数为k

在这里插入图片描述
以上图为例,具体的操作流程:假设集合里面有3个元素{x, y, z},哈希函数的个数为3(这里元素个数和哈希函数的数量没有直接关系)。

首先将位数组进行初始化,将里面每个位都设置位0。

对于集合里面的每一个元素,将元素依次通过3个哈希函数进行映射,每次映射都会产生一个哈希值,这个值对应位数组上面的一个点,然后将位数组对应的位置标记为1。

查询W元素是否存在集合中的时候,同样的方法将W通过哈希映射到位数组上的3个点。如果3个点的其中有一个点不为1,则可以判断该元素一定不存在集合中。反之,如果3个点都为1,则该元素可能存在集合中。

注意:此处不能判断该元素是否一定存在集合中,可能存在一定的误判率。可以从图中可以看到:假设某个元素通过映射对应下标为4,5,6这3个点。虽然这3个点都为1,但是很明显这3个点是不同元素经过哈希得到的位置,因此这种情况说明元素虽然不在集合中,也可能对应的都是1,这是误判率存在的原因。

java代码实现

可以看下面这段代码,也可以到 https://archive.codeplex.com/?p=bloomfilter 这个地址看开源代码

  1. import java.io.Serializable;
  2. import java.util.BitSet;
  3. import java.util.concurrent.atomic.AtomicInteger;
  4. public class BloomFileter implements Serializable {
  5. private static final long serialVersionUID = -5221305273707291280L;
  6. private final int[] seeds;
  7. private final int size;
  8. private final BitSet notebook;
  9. private final MisjudgmentRate rate;
  10. private final AtomicInteger useCount = new AtomicInteger(0);
  11. private final Double autoClearRate;
  12. /** * 默认中等程序的误判率:MisjudgmentRate.MIDDLE 以及不自动清空数据(性能会有少许提升) * * @param dataCount 预期处理的数据规模,如预期用于处理1百万数据的查重,这里则填写1000000 */
  13. public BloomFileter(int dataCount) {
  14. this(MisjudgmentRate.MIDDLE, dataCount, null);
  15. }
  16. /** * @param rate 一个枚举类型的误判率 * @param dataCount 预期处理的数据规模,如预期用于处理1百万数据的查重,这里则填写1000000 * @param autoClearRate 自动清空过滤器内部信息的使用比率,传null则表示不会自动清理, * 当过滤器使用率达到100%时,则无论传入什么数据,都会认为在数据已经存在了 * 当希望过滤器使用率达到80%时自动清空重新使用,则传入0.8 */
  17. public BloomFileter(MisjudgmentRate rate, int dataCount, Double autoClearRate) {
  18. long bitSize = rate.seeds.length * dataCount;
  19. if (bitSize < 0 || bitSize > Integer.MAX_VALUE) {
  20. throw new RuntimeException("位数太大溢出了,请降低误判率或者降低数据大小");
  21. }
  22. this.rate = rate;
  23. seeds = rate.seeds;
  24. size = (int) bitSize;
  25. notebook = new BitSet(size);
  26. this.autoClearRate = autoClearRate;
  27. }
  28. public void add(String data) {
  29. checkNeedClear();
  30. for (int i = 0; i < seeds.length; i++) {
  31. int index = hash(data, seeds[i]);
  32. setTrue(index);
  33. }
  34. }
  35. /** * 如果不存在就进行记录并返回false,如果存在了就返回true * * @param data * @return */
  36. public boolean addIfNotExist(String data) {
  37. checkNeedClear();
  38. int[] indexs = new int[seeds.length];
  39. // 先假定存在
  40. boolean exist = true;
  41. int index;
  42. for (int i = 0; i < seeds.length; i++) {
  43. indexs[i] = index = hash(data, seeds[i]);
  44. if (exist) {
  45. if (!notebook.get(index)) {
  46. // 只要有一个不存在,就可以认为整个字符串都是第一次出现的
  47. exist = false;
  48. // 补充之前的信息
  49. for (int j = 0; j <= i; j++) {
  50. setTrue(indexs[j]);
  51. }
  52. }
  53. } else {
  54. setTrue(index);
  55. }
  56. }
  57. return exist;
  58. }
  59. private void checkNeedClear() {
  60. if (autoClearRate != null) {
  61. if (getUseRate() >= autoClearRate) {
  62. synchronized (this) {
  63. if (getUseRate() >= autoClearRate) {
  64. notebook.clear();
  65. useCount.set(0);
  66. }
  67. }
  68. }
  69. }
  70. }
  71. public void setTrue(int index) {
  72. useCount.incrementAndGet();
  73. notebook.set(index, true);
  74. }
  75. private int hash(String data, int seeds) {
  76. char[] value = data.toCharArray();
  77. int hash = 0;
  78. if (value.length > 0) {
  79. for (int i = 0; i < value.length; i++) {
  80. hash = i * hash + value[i];
  81. }
  82. }
  83. hash = hash * seeds % size;
  84. // 防止溢出变成负数
  85. return Math.abs(hash);
  86. }
  87. public double getUseRate() {
  88. return (double) useCount.intValue() / (double) size;
  89. }
  90. /** * 分配的位数越多,误判率越低但是越占内存 * <p> * 4个位误判率大概是0.14689159766308 * <p> * 8个位误判率大概是0.02157714146322 * <p> * 16个位误判率大概是0.00046557303372 * <p> * 32个位误判率大概是0.00000021167340 * * @author lianghaohui */
  91. public enum MisjudgmentRate {
  92. // 这里要选取质数,能很好的降低错误率
  93. /** * 每个字符串分配4个位 */
  94. VERY_SMALL(new int[]{ 2, 3, 5, 7}),
  95. /** * 每个字符串分配8个位 */
  96. SMALL(new int[]{ 2, 3, 5, 7, 11, 13, 17, 19}), //
  97. /** * 每个字符串分配16个位 */
  98. MIDDLE(new int[]{ 2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53}), //
  99. /** * 每个字符串分配32个位 */
  100. HIGH(new int[]{ 2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97,
  101. 101, 103, 107, 109, 113, 127, 131});
  102. private int[] seeds;
  103. private MisjudgmentRate(int[] seeds) {
  104. this.seeds = seeds;
  105. }
  106. public int[] getSeeds() {
  107. return seeds;
  108. }
  109. public void setSeeds(int[] seeds) {
  110. this.seeds = seeds;
  111. }
  112. }
  113. public static void main(String[] args) {
  114. BloomFileter fileter = new BloomFileter(7);
  115. System.out.println(fileter.addIfNotExist("1111111111111"));
  116. System.out.println(fileter.addIfNotExist("2222222222222222"));
  117. System.out.println(fileter.addIfNotExist("3333333333333333"));
  118. System.out.println(fileter.addIfNotExist("444444444444444"));
  119. System.out.println(fileter.addIfNotExist("5555555555555"));
  120. System.out.println(fileter.addIfNotExist("6666666666666"));
  121. System.out.println(fileter.addIfNotExist("1111111111111"));
  122. }
  123. }

错误率估算

纯数学算法推导,公式参见:布隆过滤器 (Bloom Filter) 详解

下面给出一个直观的图:

  • m:存储比特位的数组长度(数组长度越长,元素越小,则误判几率越低)注意:m必须>n,不然当只有一个哈希函数的时候都一定会出现hash冲突
  • n:需要存储转换的元素的个数
  • K:把元素M映射在数组上哪一位为1的哈希函数的个数。 K要 <= m/n

在这里插入图片描述

参考资料

布隆过滤器(Bloom Filter)的原理和实现
布隆过滤器总结(二)原理和例子
以太坊源码深入分析(10)— 以太坊Bloom过滤器实现原理及应用场景分析
以太坊的工作原理, 干货
Bloom Filter(布隆过滤器)的概念和原理(转)
【原】布隆过滤器 (Bloom Filter) 详解

发表评论

表情:
评论列表 (有 0 条评论,411人围观)

还没有评论,来说两句吧...

相关阅读

    相关 过滤器

    布隆过滤器 什么是布隆过滤器? 布隆过滤器(Bloom Filter)是1970年由布隆提出的。它实际上是一个很长的`二进制向量`和`一系列随机映射函数`。布

    相关 过滤器

    > 布隆过滤器\[1\](Bloom Filter)是由布隆在1970年提出的。它实际上是由一个很长的二进制向量和一系列随机映射函数组成,布隆过滤器可以用于检索一个元素是否在一

    相关 过滤器

    布隆过滤器实际上就是哈希和位图的结合 它的优点:速度快并且节省空间 它的缺点:存在误判(比如存在不同的字符串可能存在相同的ASCII,这样我们在判断的时候就会出现误判)

    相关 过滤器

    布隆过滤器原理 布隆过滤器有什么用? 布隆过滤器是可以用于判断一个元素是不是在一个集合里,并且相比于其它的数据结构,布隆过滤器在空间和时间方面都有巨大的优势。 特

    相关 过滤器

    布隆过滤器介绍 > 布隆过滤器在wiki上的介绍: 布隆过滤器(Bloom Filter)是1970年由布隆提出的。它实际上是一个很长的二进制向量和一系列随机映射函数。布

    相关 过滤器

    布隆过滤器常常被用来检测某个元素是否是巨量数据集合中的成员 1、基本原理: (1)将长度为m的位数组元素全部置为0; (2)对集合S中的某个成员a,分别用k个哈希函数对其