详解布隆过滤器

什么是布隆过滤器

布隆过滤器(Bloom Filter)是一种空间高效的概率数据结构,它用于快速判断一个元素是否属于一个集合,特点是:

  • 时间高效:插入和查询时间复杂度均为 O(k)(k 为哈希函数个数,通常很小);
  • 空间高效:只需一个位数组,远小于传统哈希表或集合;
  • 允许假阳性:可能把不在集合中的元素误判为"可能存在",但绝无假阴性:如果判断"一定不存在",则绝对正确。

布隆过滤器解决了什么问题

在处理海量数据时,传统数据结构(如 HashSet、HashMap)会占用大量内存。例如:

  • 爬虫需要对已访问的 URL 去重(数亿 URL);
  • 缓存系统防止"缓存穿透"(先查 Redis 不存在再查 DB);
  • 垃圾邮件过滤、黑名单查询、字典拼写检查、大数据去重等场景。

这些场景下,如果用完整存储所有元素,内存消耗巨大(每个元素至少几十字节)。布隆过滤器用极少的内存(每个元素只需几比特)近似表示集合,以极小的误判率换取空间和速度。误判率可通过增大位数组长度 m 或调整哈希函数个数 k 来控制,通常控制在 1% 以内即可满足实际需求。

布隆过滤器实现原理

1.数据结构:

布隆过滤器由「初始值都为 0 的位图数组」和「 k 个哈希函数」两部分组成:

  • 一个长度为 m 的位数组(bit array),初始全部为 0;
  • k 个相互独立的哈希函数 h₁, h₂, ..., hₖ,每个哈希函数能将任意元素映射到 [0, m-1] 范围内。
2.操作:

插入元素 x:

计算 k 个哈希值:h₁(x), h₂(x), ..., hₖ(x)

将对应位置的 bit 全部置为 1。

查询元素 x:

计算同样的 k 个哈希值。

若所有 k 个位置的 bit 均为 1 -> 可能存在(可能假阳性);

若任意一个 位置的 bit 为 0 -> 一定不存在。

举个例子,假设有一个位图数组长度为 8,哈希函数 3 个的布隆过滤器。

在数据库写入数据 x 后,把数据 x 标记在布隆过滤器时,数据 x 会被 3 个哈希函数分别计算出 3 个哈希值,然后在对这 3 个哈希值对 8 取模,假设取模的结果为 1、4、6,然后把位图数组的第 1、4、6 位置的值设置为 1。当应用要查询数据 x 是否数据库时,通过布隆过滤器只要查到位图数组的第 1、4、6 位置的值是否全为 1,只要有一个为 0,就认为数据 x 不在数据库中。

布隆过滤器由于是基于哈希函数实现查找的,高效查找的同时存在哈希冲突的可能性,比如数据 x 和数据 y 可能都落在第 1、4、6 位置,而事实上,可能数据库中并不存在数据 y,存在误判的情况。

所以,查询布隆过滤器说数据存在,并不一定证明数据库中存在这个数据,但是查询到数据不存在,数据库中一定就不存在这个数据。

相关推荐
CheerWWW2 小时前
C++学习笔记——this关键字、对象生命周期(栈作用域)、智能指针、复制与拷贝构造函数
c++·笔记·学习
lucky九年2 小时前
GO语言模拟C++封装,继承,多态
开发语言·c++·golang
漫随流水3 小时前
c++编程:D进制的A+B(1022-PAT乙级)
数据结构·c++·算法
tankeven3 小时前
HJ159 没挡住洪水
c++·算法
charlie1145141913 小时前
嵌入式C++教程实战之Linux下的单片机编程:从零搭建 STM32 开发工具链(5):调试进阶篇 —— 从 printf 到完整 GDB 调试环境
linux·c++·单片机·学习·嵌入式·c
paeamecium3 小时前
【PAT】 - Course List for Student (25)
数据结构·c++·算法·pat考试
小黄人软件3 小时前
MFC为什么不报空指针异常 2宏定义不改源码 用替换 用Log函数替换printf等 #define printf Log 优化版底层类Log显示
c++·mfc
漫随流水4 小时前
c++编程:说反话(1009-PAT乙级)
数据结构·c++·算法
计算机安禾4 小时前
【数据结构与算法】第23篇:树、森林与二叉树的转换
c语言·开发语言·数据结构·c++·线性代数·算法·矩阵
ab1237684 小时前
C++ size() 与 length() 核心笔记
开发语言·c++·笔记