mysql布隆过滤器_布隆过滤器及其使用实例

本文介绍了布隆过滤器,一种高效、节省空间的概率型数据结构,用于判断某个元素是否存在。通过多个哈希函数将元素映射到bit数组中,支持add和isExist操作,但不支持删除。文章探讨了HashMap的问题并详细解释了布隆过滤器的工作原理,包括误判可能性。此外,还展示了如何使用布隆过滤器解决Redis缓存穿透问题的实例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、什么是布隆过滤器

本质上布隆过滤器是一种数据结构,比较巧妙的概率型数据结构(probabilistic data structure),特点是高效地插入和查询,可以用来告诉你 “某样东西一定不存在或者可能存在”。

相比于传统的 List、Set、Map 等数据结构,它更高效、占用空间更少,但是缺点是其返回的结果是概率性的,而不是确切的。

2、实现原理

HashMap 的问题

讲述布隆过滤器的原理之前,我们先思考一下,通常你判断某个元素是否存在用的是什么?应该蛮多人回答 HashMap 吧,确实可以将值映射到 HashMap 的 Key,然后可以在 O(1) 的时间复杂度内返回结果,效率奇高。但是 HashMap 的实现也有缺点,例如存储容量占比高,考虑到负载因子的存在,通常空间是不能被用满的,而一旦你的值很多例如上亿的时候,那 HashMap 占据的内存大小就变得很可观了。

还比如说你的数据集存储在远程服务器上,本地服务接受输入,而数据集非常大不可能一次性读进内存构建 HashMap 的时候,也会存在问题。

布隆过滤器数据结构

布隆过滤器是一个 bit 向量或者说 bit 数组,如图:

2a6ecb56b911

图1

如果我们要映射一个值到布隆过滤器中,我们需要使用多个不同的哈希函数生成多个哈希值,并对每个生成的哈希值指向的 bit 位置 1,例如针对值 “zhansan” 和三个不同的哈希函数分别生成了哈希值 1、4、7,则上图转变为:

2a6ecb56b911

图2

我们现在再存一个值 “lisi”,如果哈希函数返回 3、4、8 的话,则图为:

2a6ecb56b911

图3

值得注意的是,4 这个 bit 位由于两个值的哈希函数都返回了这个 bit 位,因此它被覆盖了。现在我们如果想查询 “wangwu” 这个值是否存在,哈希函数返回了 1、5、8三个值,结果我们发现 5 这个 bit 位上的值为 0,说明没有任何一个值映射到这个 bit 位上,因此我们可以很确定地说 “wangwu” 这个值不存在。而当我们需要查询 “zhansan” 这个值是否存在的话,那么哈希函数必然会返回 1、4、7,然后我们检查发现这三个 bit 位上的值均为 1,那么我们可以说 “zhansan” 存在了么?答案是不可以,只能是 “zhansan” 这个值可能存在。

这是为什么呢?答案跟简单,因为随着增加的值越来越多,被置为 1 的 bit 位也会越来越多,这样某个值 “taobao” 即使没有被存储过,但是万一哈希函数返回的三个 bit 位都被其他值置位了 1 ,那么程序还是会判断 “taobao” 这个值存在。

支持删除么

目前我们知道布隆过滤器可以支持 add 和 isExist 操作,那么 delete 操作可以么,答案是不可以,例如上图中的 bit 位 4 被两个值共同覆盖的话,一旦你删除其中一个值例如 “tencent” 而将其置位 0,那么下次判断另一个值例如 “baidu” 是否存在的话,会直接返回 false,而实际上你并没有删除它。

如何解决这个问题,答案是计数删除。但是计数删除需要存储一个数值,而不是原先的 bit 位,会增大占用的内存大小。这样的话,增加一个值就是将对应索引槽上存储的值加一,删除则是减一,判断是否存在则是看值是否大于0。

3、实例

使用布隆过滤器解决Redis缓存穿透问题

pom.xml

org.springframework.boot

spring-boot-starter-parent

2.0.1.RELEASE

org.projectlombok

lombok

org.springframework.boot

spring-boot-starter-web

org.springframework.boot

spring-boot-starter-data-redis

mysql

mysql-connector-java

org.mybatis.spring.boot

mybatis-spring-boot-starter

2.1.0

com.google.guava

guava

20.0

org.springframework.cloud

spring-cloud-dependencies

Finchley.M7

pom

import

application.yml

spring:

redis:

host: 127.0.0.1

port:6379

password: 123456

database:1

datasource:

driver-class-name: com.mysql.jdbc.Driver

url: jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf-8&useSSL=false

username: root

password: root

RedisTemplateUtils.java

@Component

public class RedisTemplateUtils {

@Resource

private RedisTemplateredisTemplate;

public void set(K k,V v){

set(k, v,null);

}

public void set(K k,V v, Long timeout){

redisTemplate.opsForValue().set(k, v);

if(timeout !=null){

redisTemplate.expire(k, timeout, TimeUnit.SECONDS);

}

}

public V get(K k){

return redisTemplate.opsForValue().get(k);

}

}

UserEntity.java

@Data

public class UserEntityimplements Serializable {

private int userId;

private StringuserName;

}

UserMapper.java

public interface UserMapper {

@Select("select userId, userName from user_t where userId=#{userId}")

UserEntity getUser(int userId);

@Select("select userId from user_t ")

List getUserIds();

}

BloomFilterInit.java 当Spring启动后初始化布隆过滤器

@Component

public class BloomFilterInit implements ApplicationRunner {

private BloomFilterbloomFilter;

@Autowired

private UserMapperuserMapper;

@Override

public void run(ApplicationArguments args)throws Exception {

List userIds =userMapper.getUserIds();

if (userIds.size() >0) {

// 0.01即错误率为1%

bloomFilter = BloomFilter.create(Funnels.integerFunnel(), userIds.size(),0.01);

for (int i =0; i < userIds.size(); i++) {

bloomFilter.put(userIds.get(i));

}

System.out.println("预热userId到布隆过滤器成功!");

}

}

public BloomFilter getIntegerBloomFilter() {

return bloomFilter;

}

}

UserController.java

@RestController

public class UserController {

@Autowired

private UserMapperuserMapper;

@Autowired

private RedisTemplateUtilsredisTemplateUtils;

@Autowired

private BloomFilterInitbloomFilterInit;

@RequestMapping("/getUser")

public UserEntity getUser(int userId){

if(!bloomFilterInit.getIntegerBloomFilter().mightContain(userId)){

System.out.println("该userId在布隆过滤器中不存在!");

return null;

}

UserEntity userEntity =redisTemplateUtils.get(userId +"");

if(userEntity !=null){

System.out.println("从Redis中返回数据!");

return userEntity;

}

System.out.println("从数据库中查询数据!");

UserEntity user =userMapper.getUser(userId);

if(user !=null){

System.out.println("将数据缓存到Redis中!");

redisTemplateUtils.set(userId+"", user);

}

return user;

}

}

App.java

@SpringBootApplication

@MapperScan("com.ttcv.bloom.mapper")

public class App {

public static void main(String[] args) {

SpringApplication.run(App.class);

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值