赞
踩
布隆过滤器是一种基于位数组(bit array)和多个哈希函数的数据结构。其核心原理是:
空间效率高:布隆过滤器通过使用位数组和哈希函数,可以在相对较小的空间内表示一个大型集合。这使得它特别适合内存受限的应用场景。
插入和查询速度快:插入和查询操作都只需要O(k)的时间复杂度(k为哈希函数的数量),非常高效。哈希函数的计算和位数组的访问都可以在常数时间内完成。
无需存储实际元素:布隆过滤器只需要存储哈希值,并不需要存储实际的元素数据,因此它能有效地节省存储空间。
适用于分布式系统:布隆过滤器可以轻松地分布在多个节点上,通过分布式哈希算法进行管理,适用于大规模分布式系统。
扩展性好:一些扩展版本的布隆过滤器,如可扩展布隆过滤器(Scalable Bloom Filter),可以动态调整大小,适应不断增长的数据集。
存在误判率:布隆过滤器有一定的误判率,即可能会误判一个不在集合中的元素为存在。误判率取决于位数组的大小、哈希函数的数量和存储的元素数量。这是由于哈希冲突产生的。
无法删除元素:基本布隆过滤器不支持删除操作,因为无法确定一个位置上的1是由哪个元素设置的。虽然计数布隆过滤器(Counting Bloom Filter)可以支持删除操作,但代价是需要更多的空间。
初始化参数选择复杂:选择合适的位数组大小和哈希函数数量是一个复杂的问题。位数组太小或哈希函数数量太少会增加误判率,而位数组太大或哈希函数数量太多则会浪费空间和时间。
不适用于动态集:基本布隆过滤器在初始化时需要确定位数组的大小,这对于元素数量动态变化的场景并不友好。可扩展布隆过滤器虽然可以动态调整大小,但实现较为复杂。
不支持元素的完整存储和检索:布隆过滤器只能判断元素是否存在于集合中,无法存储和检索元素的实际内容。
布隆过滤器在很多应用场景中都有广泛的应用:
缓存系统:在缓存系统中,布隆过滤器可以用来快速判断一个请求是否命中缓存,避免不必要的数据库查询,解决缓存穿透问题。
垃圾邮件过滤:邮件系统可以使用布隆过滤器来快速判断一封邮件是否是垃圾邮件。
网络爬虫:在网络爬虫中,布隆过滤器可以用来记录已经访问过的URL,避免重复抓取。
数据库去重:在大规模数据处理中,布隆过滤器可以用来快速判断一个记录是否已经存在,避免重复存储。
分布式系统:在分布式系统中,布隆过滤器可以用来快速判断一个数据是否存在于某个节点上,提高查询效率。
常用的几种有单体项目下,使用Guava包下的BloomFilter,分布式下使用Redission的RBloomFilter,这些都是写好的布隆过滤器,接下来将基于redis和jedis实现一个手写的分布式布隆过滤器
分布式布隆过滤器在大规模分布式系统中应用广泛,它的实现主要涉及以下几个方面:
一致性哈希是一种用于分布式系统的哈希算法,能够有效地应对节点动态加入和退出的情况。它通过将所有节点和数据哈希到一个环上来实现数据的分布。主要包含以下步骤:
下面是用Java和Jedis实现的分布式布隆过滤器示例。我们使用一致性哈希来分配数据,并用Redis存储位数组。
1. 一致性哈希实现
- import java.util.SortedMap;
- import java.util.TreeMap;
-
- public class ConsistentHashing {
- private final SortedMap<Integer, String> circle = new TreeMap<>();
- private final int replicas;
-
- public ConsistentHashing(int replicas) {
- this.replicas = replicas;
- }
-
- public void addNode(String node) {
- for (int i = 0; i < replicas; i++) {
- circle.put((node + i).hashCode(), node);
- }
- }
-
- public void removeNode(String node) {
- for (int i = 0; i < replicas; i++) {
- circle.remove((node + i).hashCode());
- }
- }
-
- public String getNode(String key) {
- if (circle.isEmpty()) {
- return null;
- }
- int hash = key.hashCode();
- if (!circle.containsKey(hash)) {
- SortedMap<Integer, String> tailMap = circle.tailMap(hash);
- hash = tailMap.isEmpty() ? circle.firstKey() : tailMap.firstKey();
- }
- return circle.get(hash);
- }
- }
2. 分布式布隆过滤器实现
- import redis.clients.jedis.Jedis;
- import java.nio.charset.StandardCharsets;
- import com.google.common.hash.Hashing;
-
- public class DistributedBloomFilter {
- private ConsistentHashing consistentHashing;
- private int size;
- private int numHashFunctions;
-
- public DistributedBloomFilter(int replicas, int size, int numHashFunctions) {
- this.consistentHashing = new ConsistentHashing(replicas);
- this.size = size;
- this.numHashFunctions = numHashFunctions;
- }
-
- public void addNode(String host, int port) {
- consistentHashing.addNode(host + ":" + port);
- }
-
- public void removeNode(String host, int port) {
- consistentHashing.removeNode(host + ":" + port);
- }
-
- private static int[] getHashes(String value, int numHashes, int maxSize) {
- int[] hashes = new int[numHashes];
- for (int i = 0; i < numHashes; i++) {
- hashes[i] = Math.abs(Hashing.murmur3_128(i).hashString(value, StandardCharsets.UTF_8).asInt() % maxSize);
- }
- return hashes;
- }
-
- private Jedis getJedisClient(String value) {
- // 使用一致性哈希算法找到合适的节点
- String node = consistentHashing.getNode(value);
- // 解析节点信息并创建Jedis客户端实例
- String[] parts = node.split(":");
- return new Jedis(parts[0], Integer.parseInt(parts[1]));
- }
-
- public void add(String value) {
- // 计算哈希值
- int[] hashes = getHashes(value, numHashFunctions, size);
- try (Jedis jedis = getJedisClient(value)) {
- // 设置位数组的对应位置
- for (int hash : hashes) {
- jedis.setbit("bloom_filter", hash, true);
- }
- }
- }
-
- public boolean contains(String value) {
- // 计算哈希值
- int[] hashes = getHashes(value, numHashFunctions, size);
- try (Jedis jedis = getJedisClient(value)) {
- // 查询位数组的对应位置
- for (int hash : hashes) {
- if (!jedis.getbit("bloom_filter", hash)) {
- return false;
- }
- }
- }
- return true;
- }
-
- public static void main(String[] args) {
- // 创建布隆过滤器实例
- DistributedBloomFilter bloomFilter = new DistributedBloomFilter(3, 1000, 5);
-
- // 添加Redis节点
- bloomFilter.addNode("localhost", 6379);
- bloomFilter.addNode("localhost", 6380);
- bloomFilter.addNode("localhost", 6381);
-
- // 插入元素
- bloomFilter.add("apple");
- bloomFilter.add("banana");
-
- // 查询元素
- System.out.println(bloomFilter.contains("apple")); // 输出: true
- System.out.println(bloomFilter.contains("banana")); // 输出: true
- System.out.println(bloomFilter.contains("cherry")); // 输出: false
- }
- }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。