Redis解决穿透击穿问题时使用的布隆过滤器知识点

上一篇 <<<Redis缓存的穿透、击穿和雪崩效应
下一篇 >>>Redis与MySQL的数据同步解决方案


1.布隆过滤器介绍

布隆过滤器是一个很长的二进制向量和一系列随机映射函数,适用于判断某个数据在集合中是否存在,会存在误识别。
优点是空间效率和查询时间都比一般的算法要好的多
缺点是有一定的误识别率和删除困难。

2.布隆过滤器使用场景

客户端--布隆过滤器(hashmap)-redis缓存--DB数据库
a、在程序启动时,将redis的所有key先缓存预热(加载)到布隆过滤器中,也可以用hashmap,但布隆过滤器的性能会比hashmap快很多
b、客户端请求的时候,先经过布隆过滤器,判断key是否存在,不存在的话,直接返回,可以解决redis的穿透和击穿。
c、布隆过滤器误判经过redis里,也不会造成原先大批量的涌入,这是可以接受的
d、如果redis的key有所变更,让布隆过滤器重新缓存预热---可解决删除问题

3.布隆过滤器存在的问题

3.1 误判

上图Jarye2本身在二进制向量表中不存在,由于hash值和其他碰撞,导致以为存在。
解决方式:把误判概率设置的足够小,但会导致向量表会大很多。

3.2 删除困难

假设上图中把Jarye2删了,会把向量地址8 13的值设置为0,导致原先应该命中的Jarye1无法命中
解决方式:缓存重新预热。

4.布隆过滤器demo示例

  • 依赖引入
<dependency>
    <groupId>com.google.guava</groupId>
    <artifactId>guava</artifactId>
    <version>22.0</version>
</dependency>
  • 测试demo
/**
 * 测试demo
 */
public class BlongTest {
    /**
     * 在布隆中存放100万条数据
     */
    private static Integer size = 1000000;

    public static void main(String[] args) {
        /**
         * 最后参数为误判率,必须要>0.0
         * 误判率是3% 100W的数据,二巷数组长度为730W
         * 误判率是1% 100W的数据,二巷数组长度为960W
         * 综合效率和准确率,建议值是1%
         *
         */
        BloomFilter<Integer> integerBloomFilter = BloomFilter.create(Funnels.integerFunnel(), size, 0.01);
        for (int i = 0; i < size; i++) {
            integerBloomFilter.put(i);
        }
        // 从布隆中查询数据是否存在
        ArrayList<Integer> strings = new ArrayList<>();
        for (int j = size; j < size + 10000; j++) {
            if (integerBloomFilter.mightContain(j)) {
                strings.add(j);
            }
        }
        System.out.println("误判数量:" + strings.size());

    }
}

5.基于布隆过滤器解决Redis击穿问题

@RequestMapping("/getOrder")
public OrderEntity getOrder(Integer orderId) {
    if (integerBloomFilter != null) {
        if (!integerBloomFilter.mightContain(orderId)) {
            System.out.println("从布隆过滤器中检测到该key不存在");
            return null;
        }
    }
    // 1.先查询Redis中数据是否存在
    OrderEntity orderRedisEntity = (OrderEntity) redisTemplateUtils.getObject(orderId + "");
    if (orderRedisEntity != null) {
        System.out.println("直接从Redis中返回数据");
        return orderRedisEntity;
    }
    // 2. 查询数据库的内容
    System.out.println("从DB查询数据");
    OrderEntity orderDBEntity = orderMapper.getOrderById(orderId);
    if (orderDBEntity != null) {
        System.out.println("将Db数据放入到Redis中");
        redisTemplateUtils.setObject(orderId + "", orderDBEntity);
    }
    return orderDBEntity;
}

@RequestMapping("/dbToBulong")
public String dbToBulong() {
    List<Integer> orderIds = orderMapper.getOrderIds();
    integerBloomFilter = BloomFilter.create(Funnels.integerFunnel(), orderIds.size(), 0.01);
    for (int i = 0; i < orderIds.size(); i++) {
        integerBloomFilter.put(orderIds.get(i));
    }
    return "success";
}

推荐阅读:
<<<分布式缓存与本地缓存的区别
<<<Ehcache基础知识
<<<SpringBoot整合Ehcache
<<<Redis的5种数据类型
<<<Redis存放实体对象的方式及区别
<<<Redis的应用场景汇总
<<<Redis高效及线程安全的真正原因
<<<Redis为啥要分为16个库
<<<RDB和AOF持久化方式的区别
<<<Redis与数据库的一致性解决方案
<<<SpringBoot整合Redis的注解版本完成数据缓存
<<<Redis的淘汰策略
<<<Redis的事务操作(Mult和Watch)知识点
<<<Redis的过期机制使用场景示例
<<<Redis实现分布式锁的原理分析
<<<Redis分布式锁的实现代码示例
<<<使用Redisson工具实现分布式锁
<<<Redis集群模式之主从复制原理及存在的缺陷
<<<Redis集群模式之哨兵模式
<<<Redis集群模式之Cluster去中心化分片集群
<<<Linux环境下安装单机Redis
<<<Redis Cluster集群环境搭建
<<<Redis Cluster如何动态扩容与缩容
<<<Redis Cluster主从节点自动切换
<<<Redis集群模式的类型和缺陷汇总
<<<Redis缓存的穿透、击穿和雪崩效应
<<<Redis与MySQL的数据同步解决方案
<<<阿里云的Canal框架实现Redis与Mysql同步原理及代码示例
<<<阿里云的Canal框架配置
<<<Redis官方提出的redlock分布式锁
<<<Redis的调优设置
<<<Redis常见问题汇总

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容