上周在工作中遇到了一个问题场景,即查询商品的配件信息时(商品:配件为1:N的关系),如若商品并未配置配件信息,则查数据库为空,且不会加入缓存,这就会导致,下次在查询同样商品的配件时,由于缓存未命中,则仍旧会查底层数据库,所以缓存就一直未起到应有的作用,当并发流量大时,会很容易把DB打垮。

缓存穿透问题

缓存穿透是指查询一个根本不存在的数据,缓存层和存储层都不会命中,通常出于容错的考虑,如果从存储层查不到数据则不写入缓存层。
一般对于未命中的数据我们是按照如下方式进行处理的:

1.缓存层不命中。
2.存储层不命中,不将空结果写回缓存。
3.返回空结果。

  1. /**
  2. * 缓存穿透问题:
  3. * 在数据库层没有查到数据,未存入缓存,
  4. * 则下次查询同样的数据时,还会查库。
  5. *
  6. * @param id
  7. * @return
  8. */
  9. private Object getObjectById(Integer id) {
  10. // 从缓存中获取数据
  11. Object cacheValue = cache.get(id);
  12. if (cacheValue != null) {
  13. return cacheValue;
  14. }
  15. // 从数据库中获取
  16. Object storageValue = storage.get(id);
  17. // 如果这里按照id查询DB为空,那么便会出现缓存穿透
  18. if (storageValue != null) {
  19. cache.set(id, storageValue);
  20. }
  21. return storageValue;
  22. }

缓存穿透将导致不存在的数据每次请求都要到存储层去查询,失去了缓存保护后端存储的意义。
缓存穿透问题可能会使后端存储负载加大,由于很多后端存储不具备高并发性,甚至可能造成后端存储宕掉。

方案一:缓存空对象

  1. /**
  2. * 缓存空对象:
  3. * 此种方式存在漏洞,不经过判断就直接将Null对象存入到缓存中,
  4. * 如果恶意制造不存在的id那么,缓存中的键值就会很多,恶意攻击时,很可能会被打爆,所以需设置较短的过期时间。
  5. *
  6. * @param id
  7. * @return
  8. */
  9. public Object getObjectInclNullById(Integer id) {
  10. // 从缓存中获取数据
  11. Object cacheValue = cache.get(id);
  12. // 缓存为空
  13. if (cacheValue != null) {
  14. // 从数据库中获取
  15. Object storageValue = storage.get(key);
  16. // 缓存空对象
  17. cache.set(key, storageValue);
  18. // 如果存储数据为空,需要设置一个过期时间(300秒)
  19. if (storageValue == null) {
  20. // 必须设置过期时间,否则有被攻击的风险
  21. cache.expire(key, 60 * 5);
  22. }
  23. return storageValue;
  24. }
  25. return cacheValue;
  26. }

缓存空对象会有一个必须考虑的问题:

空值做了缓存,意味着缓存层中存了更多的键,需要更多的内存空间(如果是攻击,问题更严重),比较有效的方法是针对这类数据设置一个较短的过期时间,让其自动剔除。

方案二:布隆过滤器拦截

布隆过滤器介绍

概念:

布隆过滤器(英语:Bloom Filter)是1970年由布隆提出的。它实际上是一个很长的二进制向量和一系列随机映射函数。布隆过滤器可以用于检索一个元素是否在一个集合中。它的优点是空间效率和查询时间都远远超过一般的算法,缺点是有一定的误识别率和删除困难。

如果想判断一个元素是不是在一个集合里,一般想到的是将集合中所有元素保存起来,然后通过比较确定。链表、树、散列表(又叫哈希表,Hash table)等等数据结构都是这种思路。但是随着集合中元素的增加,我们需要的存储空间越来越大。同时检索速度也越来越慢,上述三种结构的检索时间复杂度分别为 O(n),O(log n),O(n/k)

布隆过滤器的原理是,当一个元素被加入集合时,通过K个散列函数将这个元素映射成一个位数组中的K个点,把它们置为1。检索时,我们只要看看这些点是不是都是1就(大约)知道集合中有没有它了:如果这些点有任何一个0,则被检元素一定不在;如果都是1,则被检元素很可能在。这就是布隆过滤器的基本思想。

示例:

google guava包下有对布隆过滤器的封装,BloomFilter。

  1. import com.google.common.hash.BloomFilter;
  2. import com.google.common.hash.Funnels;
  3.  
  4. public class BloomFilterTest {
  5.  
  6. // 初始化一个能够容纳10000个元素且容错率为0.01布隆过滤器
  7. private static final BloomFilter<Integer> bloomFilter = BloomFilter.create(Funnels.integerFunnel(), 10000, 0.01);
  8.  
  9. /**
  10. * 初始化布隆过滤器
  11. */
  12. private static void initLegalIdsBloomFilter() {
  13. // 初始化10000个合法Id并加入到过滤器中
  14. for (int legalId = 0; legalId < 10000; legalId++) {
  15. bloomFilter.put(legalId);
  16. }
  17. }
  18.  
  19. /**
  20. * id是否合法有效,即是否在过滤器中
  21. *
  22. * @param id
  23. * @return
  24. */
  25. public static boolean validateIdInBloomFilter(Integer id) {
  26. return bloomFilter.mightContain(id);
  27. }
  28.  
  29. public static void main(String[] args) {
  30. // 初始化过滤器
  31. initLegalIdsBloomFilter();
  32. // 误判个数
  33. int errorNum=0;
  34. // 验证从10000个非法id是否有效
  35. for (int id = 10000; id < 20000; id++) {
  36. if (validateIdInBloomFilter(id)){
  37. // 误判数
  38. errorNum++;
  39. }
  40. }
  41. System.out.println("judge error num is : " + errorNum);
  42. }
  43. }

布隆过滤器拦截

设置过期时间,让其自动过期失效,这种在很多时候不是最佳的实践方案。

我们可以提前将真实正确的商品Id,在添加完成之后便加入到过滤器当中,每次再进行查询时,先确认要查询的Id是否在过滤器当中,如果不在,则说明Id为非法Id,则不需要进行后续的查询步骤了。

  1. /**
  2. * 防缓存穿透的:布隆过滤器
  3. *
  4. * @param id
  5. * @return
  6. */
  7. public Object getObjectByBloom(Integer id) {
  8. // 判断是否为合法id
  9. if (!bloomFilter.mightContain(id)) {
  10. // 非法id,则不允许继续查库
  11. return null;
  12. } else {
  13. // 从缓存中获取数据
  14. Object cacheValue = cache.get(id);
  15. // 缓存为空
  16. if (cacheValue == null) {
  17. // 从数据库中获取
  18. Object storageValue = storage.get(id);
  19. // 缓存空对象
  20. cache.set(id, storageValue);
  21. }
  22. return cacheValue;
  23. }
  24. } 

参考书籍:《Redis开发与运维》  

Redis缓存穿透问题及解决方案的更多相关文章

  1. Redis缓存穿透和缓存雪崩以及解决方案

    Redis缓存穿透和缓存雪崩以及解决方案 Redis缓存穿透和缓存雪崩以及解决方案缓存穿透解决方案布隆过滤缓存空对象比较缓存雪崩解决方案保证缓存层服务高可用性依赖隔离组件为后端限流并降级数据预热缓存并 ...

  2. Redis缓存穿透、缓存雪崩、redis并发问题 并发竞争key的解决方案 (阿里)

    阿里的人问我 缓存雪崩(大量数据在同一时间过期了)了如何处理,缓存击穿了如何处理,回答的很烂,做了总结: 把redis作为缓存使用已经是司空见惯,但是使用redis后也可能会碰到一系列的问题,尤其是数 ...

  3. redis缓存穿透穿透解决方案-布隆过滤器

    redis缓存穿透穿透解决方案-布隆过滤器 我们先来看一段代码 cache_key = "id:1" cache_value = GetValueFromRedis(cache_k ...

  4. 预防Redis缓存穿透、缓存雪崩解决方案

    最近面试中遇到redis缓存穿透.缓存雪崩等问题,特意了解下. redis缓存穿透: 缓存穿透是指用户查询数据,在数据库没有,自然在缓存中也不会有.这样就导致用户查询的时候,在缓存中找不到,每次都要去 ...

  5. redis缓存穿透,缓存击穿,缓存雪崩

    概念解释 redis 缓存穿透 key对应的数据在数据源并不存在,每次针对此key的请求从缓存获取不到,请求都会到数据源,从而可能压垮数据源.比如用一个不存在的用户id获取用户信息,不论缓存还是数据库 ...

  6. 【干货!!】三句话搞懂 Redis 缓存穿透、击穿、雪崩

    前言 如何有效的理解并且区分 Reids 穿透.击穿和雪崩之间的区别,一直以来都挺困扰我的.特别是穿透和击穿,过一段时间就稀里糊涂的分不清了. 为了有效的帮助笔者自己,以及拥有同样烦恼的朋友们区分这三 ...

  7. redis与mysql性能对比、redis缓存穿透、缓存雪崩

    写在开始 redis是一个基于内存hash结构的缓存型db.其优势在于速读写能力碾压mysql.由于其为基于内存的db所以存储数据量是受限的. redis性能 redis读写性能测试redis官网测试 ...

  8. Redis 缓存穿透

    Redis 缓存穿透 https://www.cnblogs.com/jiekzou/p/9212114.html 场景描述:我们在项目中使用缓存通常都是先检查缓存中是否存在,如果存在直接返回缓存内容 ...

  9. Redis缓存穿透,缓存击穿,缓存雪崩,热点Key

    导读 使用Redis难免会遇到Redis缓存穿透,缓存击穿,缓存雪崩,热点Key的问题.有些同学可能只是会用Redis来存取,基本都是用项目里封装的工具类来操作.但是作为开发,我们使用Redis时可能 ...

随机推荐

  1. Android MediaScanner

    一.MediaScanner 的使用 1)Intent.ACTION_MEDIA_SCANNER_SCAN_FILE:扫描指定文件 public void scanFileAsync(Context ...

  2. C语言位操作--逻辑运算符组合

    假设读者熟悉普通代数与布尔代数,下面是部分常见的涉及到加法.减法与逻辑运算符的组合: a.        -x=~x+1 b.           =~(x-1) c.        ~x=-x-1 ...

  3. Docker 容器管理:rancher

    Rancher:https://www.cnrancher.com/ 是一个开源的企业级全栈化容器部署及管理平台. 定位上和 K8s 比较接近,都是通过 web 界面赋予完全的 docker 服务编排 ...

  4. cocoa 的大招(KVC的几点强大应用记录)

    1.利用KVC可以修改系统的只读变量 简单的KVC和我们平常通过一个“类名.属性”赋值一般: teacher.name = @"灭绝师太"; [teacher setValue:@ ...

  5. 3种方法教你PS快速去掉水印

    方法一:使用选框工具 勾选水印部分: 按住Shift+f5选择内容识别: 然后 ctrl+d 取消选择,水印就去掉了 PS:其实这个方法有个快捷办法,直接使用选框工具选中之后,按Delete就可以弹出 ...

  6. 【转载】51单片机data,bdata,idata,xdata使用注意事项

    "51单片机编程在不同内存空间data xdata bdata定义变量的注意事项": 关键词:51 单片机 编程 不同 内存空间 data xdatabdata 定义 变量 注意事 ...

  7. uid列表来讲讲我是如何利用php数组进行排重的

    经常接到要对网站的会员进行站内信.手机短信.email进行群发信息的通知,用户列表一般由别的同事提供,当中难免会有重复,为了避免重复发送,所以我在进行发送信息前要对他们提供的用户列表进行排重. 假如得 ...

  8. HUSTM 1601 - Shepherd

    题目描述 Hehe keeps a flock of sheep, numbered from 1 to n and each with a weight wi. To keep the sheep ...

  9. HDU-1394 Minimum Inversion Number(线段树求逆序数)

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Ot ...

  10. Linux:32/64位程序(应用程序、共享库、内核模块)

    摘要: Linux系统区分32/64位,相应地,应用程序.共享库和内核模块也区分32/64位. 本文以Ubuntu系统为例,介绍如何编译和使用32/64位的应用程序.共享库和内核模块. 1. 应用程序 ...