爆款云主机2核4G限时秒杀,88元/年起!
查看详情

活动

天翼云最新优惠活动,涵盖免费试用,产品折扣等,助您降本增效!
热门活动
  • 618智算钜惠季 爆款云主机2核4G限时秒杀,88元/年起!
  • 免费体验DeepSeek,上天翼云息壤 NEW 新老用户均可免费体验2500万Tokens,限时两周
  • 云上钜惠 HOT 爆款云主机全场特惠,更有万元锦鲤券等你来领!
  • 算力套餐 HOT 让算力触手可及
  • 天翼云脑AOne NEW 连接、保护、办公,All-in-One!
  • 中小企业应用上云专场 产品组合下单即享折上9折起,助力企业快速上云
  • 息壤高校钜惠活动 NEW 天翼云息壤杯高校AI大赛,数款产品享受线上订购超值特惠
  • 天翼云电脑专场 HOT 移动办公新选择,爆款4核8G畅享1年3.5折起,快来抢购!
  • 天翼云奖励推广计划 加入成为云推官,推荐新用户注册下单得现金奖励
免费活动
  • 免费试用中心 HOT 多款云产品免费试用,快来开启云上之旅
  • 天翼云用户体验官 NEW 您的洞察,重塑科技边界

智算服务

打造统一的产品能力,实现算网调度、训练推理、技术架构、资源管理一体化智算服务
智算云(DeepSeek专区)
科研助手
  • 算力商城
  • 应用商城
  • 开发机
  • 并行计算
算力互联调度平台
  • 应用市场
  • 算力市场
  • 算力调度推荐
一站式智算服务平台
  • 模型广场
  • 体验中心
  • 服务接入
智算一体机
  • 智算一体机
大模型
  • DeepSeek-R1-昇腾版(671B)
  • DeepSeek-R1-英伟达版(671B)
  • DeepSeek-V3-昇腾版(671B)
  • DeepSeek-R1-Distill-Llama-70B
  • DeepSeek-R1-Distill-Qwen-32B
  • Qwen2-72B-Instruct
  • StableDiffusion-V2.1
  • TeleChat-12B

应用商城

天翼云精选行业优秀合作伙伴及千余款商品,提供一站式云上应用服务
进入甄选商城进入云市场创新解决方案
办公协同
  • WPS云文档
  • 安全邮箱
  • EMM手机管家
  • 智能商业平台
财务管理
  • 工资条
  • 税务风控云
企业应用
  • 翼信息化运维服务
  • 翼视频云归档解决方案
工业能源
  • 智慧工厂_生产流程管理解决方案
  • 智慧工地
建站工具
  • SSL证书
  • 新域名服务
网络工具
  • 翼云加速
灾备迁移
  • 云管家2.0
  • 翼备份
资源管理
  • 全栈混合云敏捷版(软件)
  • 全栈混合云敏捷版(一体机)
行业应用
  • 翼电子教室
  • 翼智慧显示一体化解决方案

合作伙伴

天翼云携手合作伙伴,共创云上生态,合作共赢
天翼云生态合作中心
  • 天翼云生态合作中心
天翼云渠道合作伙伴
  • 天翼云代理渠道合作伙伴
天翼云服务合作伙伴
  • 天翼云集成商交付能力认证
天翼云应用合作伙伴
  • 天翼云云市场合作伙伴
  • 天翼云甄选商城合作伙伴
天翼云技术合作伙伴
  • 天翼云OpenAPI中心
  • 天翼云EasyCoding平台
天翼云培训认证
  • 天翼云学堂
  • 天翼云市场商学院
天翼云合作计划
  • 云汇计划
天翼云东升计划
  • 适配中心
  • 东升计划
  • 适配互认证

开发者

开发者相关功能入口汇聚
技术社区
  • 专栏文章
  • 互动问答
  • 技术视频
资源与工具
  • OpenAPI中心
开放能力
  • EasyCoding敏捷开发平台
培训与认证
  • 天翼云学堂
  • 天翼云认证
魔乐社区
  • 魔乐社区

支持与服务

为您提供全方位支持与服务,全流程技术保障,助您轻松上云,安全无忧
文档与工具
  • 文档中心
  • 新手上云
  • 自助服务
  • OpenAPI中心
定价
  • 价格计算器
  • 定价策略
基础服务
  • 售前咨询
  • 在线支持
  • 在线支持
  • 工单服务
  • 建议与反馈
  • 用户体验官
  • 服务保障
  • 客户公告
  • 会员中心
增值服务
  • 红心服务
  • 首保服务
  • 客户支持计划
  • 专家技术服务
  • 备案管家

了解天翼云

天翼云秉承央企使命,致力于成为数字经济主力军,投身科技强国伟大事业,为用户提供安全、普惠云服务
品牌介绍
  • 关于天翼云
  • 智算云
  • 天翼云4.0
  • 新闻资讯
  • 天翼云APP
基础设施
  • 全球基础设施
  • 信任中心
最佳实践
  • 精选案例
  • 超级探访
  • 云杂志
  • 分析师和白皮书
  • 天翼云·创新直播间
市场活动
  • 2025智能云生态大会
  • 2024智算云生态大会
  • 2023云生态大会
  • 2022云生态大会
  • 天翼云中国行
天翼云
  • 活动
  • 智算服务
  • 产品
  • 解决方案
  • 应用商城
  • 合作伙伴
  • 开发者
  • 支持与服务
  • 了解天翼云
      • 文档
      • 控制中心
      • 备案
      • 管理中心

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      首页 知识中心 存储 文章详情页

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2023-06-07 07:27:35 阅读次数:117

      java,redis,缓存

       

      2.1 什么是缓存?

      前言:什么是缓存?

      就像自行车,越野车的避震器《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      举个例子:越野车,山地自行车,都拥有"避震器",防止车体加速后因惯性,在酷似"U"字母的地形上飞跃,硬着陆导致的损害,像个弹簧一样;

      同样,实际开发中,系统也需要"避震器",防止过高的数据访问猛冲系统,导致其操作线程无法及时处理信息而瘫痪;

      这在实际开发中对企业讲,对产品口碑,用户评价都是致命的;所以企业非常重视缓存技术;

      缓存(Cache),就是数据交换的缓冲区,俗称的缓存就是缓冲区内的数据,一般从数据库中获取,存储于本地代码(例如:

      例1:Static final ConcurrentHashMap<K,V> map = new ConcurrentHashMap<>(); 本地用于高并发
      
      例2:static final Cache<K,V> USER_CACHE = CacheBuilder.newBuilder().build(); 用于redis等缓存
      
      例3:Static final Map<K,V> map =  new HashMap(); 本地缓存
      

      由于其被Static修饰,所以随着类的加载而被加载到内存之中,作为本地缓存,由于其又被final修饰,所以其引用(例3:map)和对象(例3:new HashMap())之间的关系是固定的,不能改变,因此不用担心赋值(=)导致缓存失效;

      2.1.1 为什么要使用缓存

      一句话:因为==速度快,好用==

      缓存数据存储于代码中,而代码运行在内存中,内存的读写性能远高于磁盘,缓存可以大大降低用户访问并发量带来的服务器读写压力

      实际开发过程中,企业的数据量,少则几十万,多则几千万,这么大数据量,如果没有缓存来作为"避震器",系统是几乎撑不住的,所以企业会大量运用到缓存技术;

      但是缓存也会增加代码复杂度和运营的成本:

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.1.2 如何使用缓存

      实际开发中,会构筑多级缓存来使系统运行速度进一步提升,例如:本地缓存与redis中的缓存并发使用

      浏览器缓存:主要是存在于浏览器端的缓存

      **应用层缓存:**可以分为tomcat本地缓存,比如之前提到的map,或者是使用redis作为缓存

      **数据库缓存:**在数据库中有一片空间是 buffer pool,增改查数据都会先加载到mysql的缓存中

      **CPU缓存:**当代计算机最大的问题是 cpu性能提升了,但内存读写速度没有跟上,所以为了适应当下的情况,增加了cpu的L1,L2,L3级的缓存

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.2 添加商户缓存

      在我们查询商户信息时,我们是直接操作从数据库中去进行查询的,大致逻辑是这样,直接查询数据库那肯定慢咯,所以我们需要增加缓存

      @GetMapping("/{id}")
      public Result queryShopById(@PathVariable("id") Long id) {
          //这里是直接查询数据库
          return shopService.queryById(id);
      }
      

      2.2.1 、缓存模型和思路

      标准的操作方式就是查询数据库之前先查询缓存,如果缓存数据存在,则直接从缓存中返回,如果缓存数据不存在,再查询数据库,然后将数据存入redis。

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.2.2、代码如下

      代码思路:如果缓存有,则直接返回,如果缓存不存在,则查询数据库,然后存入redis。

      @Override
      public Result queryById(Long id) {
          // 1.从查询Redis中是否有数据
          String shopJson = stringRedisTemplate.opsForValue().get(RedisConstants.CACHE_SHOP_KEY + id);
          // 2.如果有则直接返回
          if (StrUtil.isNotBlank(shopJson)) {
              Shop shop = JSONUtil.toBean(shopJson, Shop.class);
              return Result.ok(shop);
          }
      
          // 3.如果没有,就去查数据库
          Shop shop = this.baseMapper.selectById(id);
          // 4.如果没找到则返回错误信息
          if(shop==null){
              return Result.fail("店铺不存在~");
          }
      
          // 5.如果查到了就加入到Redis,并返回
          stringRedisTemplate.opsForValue().set(RedisConstants.CACHE_SHOP_KEY+id, JSONUtil.toJsonStr(shop));
          return Result.ok(shop);
      }
      

      2.2.3、练习:为店铺类型接口添加缓存

      店铺类型在首页和其它多个页面都会用到,如图:

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      需求:修改ShopTypeController中的queryTypeList方法,添加查询缓存

      @Override
      public Result queryTypeList() {
          // 1.先去Redis查缓存首页缓存数据
          String shopTypeListJson = stringRedisTemplate.opsForValue().get(RedisConstants.CACHE_SHOP_TYPE_KEY);
      
          // 2.如果为不为空,则直接返回
          if (StrUtil.isNotBlank(shopTypeListJson)) {
              return Result.ok(JSONUtil.toList(shopTypeListJson,ShopType.class));
          }
      
          // 3.如果为空,去查询数据库
          List <ShopType> shopTypeList = this.query().orderByDesc("sort").list();
      
          // 4.如果查询为空,则直接返回错误信息
          if (shopTypeList == null || shopTypeList.size() == 0) {
              return Result.fail("商品类型查询失败!");
          }
      
          // 5.如果不为空,则把数据存入到Redis,并返回结果 (这里可以使用List,String等结构)
          stringRedisTemplate.opsForValue().set(RedisConstants.CACHE_SHOP_TYPE_KEY,JSONUtil.toJsonStr(shopTypeList));
          return Result.ok(shopTypeList);
      }
      

      2.3 缓存更新策略

      缓存更新是redis为了节约内存而设计出来的一个东西,主要是因为内存数据宝贵,当我们向redis插入太多数据,此时就可能会导致缓存中的数据过多,所以redis会对部分数据进行更新,或者把他叫为淘汰更合适。

      **内存淘汰:**redis自动进行,当redis内存达到咱们设定的max-memery的时候,会自动触发淘汰机制,淘汰掉一些不重要的数据(可以自己设置策略方式)

      **超时剔除:**当我们给redis设置了过期时间ttl之后,redis会将超时的数据进行删除,方便咱们继续使用缓存

      **主动更新:**我们可以手动调用方法把缓存删掉,通常用于解决缓存和数据库不一致问题

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.3.1 、数据库缓存不一致解决方案

      由于我们的缓存的数据源来自于数据库,而数据库的数据是会发生变化的,因此,如果当数据库中数据发生变化,而缓存却没有同步,此时就会有一致性问题存在,其后果是:

      用户使用缓存中的过时数据,就会产生类似多线程数据安全问题,从而影响业务,产品口碑等;怎么解决呢?有如下几种方案

      • Cache Aside Pattern 人工编码方式:缓存调用者在更新完数据库后再去更新缓存,也称之为双写方案

        • 特点:需要依靠程序员在CRUD之外进行额外的代码处理!
      • Read/Write Through Pattern : 由系统本身完成,数据库与缓存的问题交由系统本身去处理

        • 如:Canel + Kafaka 搭建的服务,但是维护和开发成本较高!
      • Write Behind Caching Pattern :调用者只操作缓存,其他线程去异步处理数据库,实现最终一致

        • 优点:调用者多次操作缓存,经过一段时间线程只需要把最后一次更新后的结果同步到数据库,效率极高
        • 缺点:维护这个异步线程比较困难; 如果在一段时间内有大量更新,但是还没有触发异步线程的更新,就会导致数据库和缓存数据差别很大; 缓存宕机 会导致 数据丢失,可靠性较差

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.3.2 、双写方案存在的问题

      综合考虑使用方案一,可控性比较高~ (实际企业中用的也比较多)

      操作缓存和数据库时有三个问题需要考虑:

      1、删除缓存还是更新缓存?

      • 更新缓存:每次更新数据库都更新缓存,无效写操作较多 ×
      • 删除缓存:更新数据库时让缓存失效,查询时再更新缓存 √

      如果采用第一个方案,那么假设我们每次操作数据库后,都操作缓存,但是中间如果没有人查询,那么这个更新动作实际上只有最后一次生效,中间的更新动作意义并不大,我们可以把缓存删除,等待再次查询时,将缓存中的数据加载出来

      2、如何保证缓存与数据库的操作的同时成功或失败?

      • 单体系统,将缓存与数据库操作放在一个事务
      • 分布式系统,利用TCC等分布式事务方案

      3、先操作缓存还是先操作数据库?

      • 先删除缓存,再操作数据库 ×
      • 先操作数据库,再删除缓存 √

      在不考虑并发的情况下,都是可以的。但是在并发场景下,我们应当是先操作数据库,再删除缓存,我们简单分析一下:

      • 如果你选择第一种方案,如左图所示:在两个线程(线程1是更新线程,线程2是查询线程)并发来访问时,假设线程1先来,他先把缓存删了,此时线程2过来,他查询缓存数据并不存在,此时他写入缓存,当他写入缓存后,线程1再执行更新动作时,实际上写入缓存的是旧的数据,新的数据被旧数据覆盖了。

      • 如果你选择第二种方案,如右图所示,两个线程(线程1是查询线程,线程2是更新线程)并发来访问时,假如此时缓存是被删除的状态,线程1过来了,发现未命中,就去数据库擦汗寻数据了,刚查询完还没写入缓存!!! 此时线程2过来了,就更新数据库了(下面删除缓存这一步就不用做了)。然后,线程1又抢到了时间片,继续进行写入缓存操作,就会导致写入缓存的仍是旧数据

      既然我们发现两种方案都有并发修改的风险,但是为啥选择方案二呢?

      因为 查询操作时间非常短,容易产生并发风险概率较低~

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      总结:缓存更新策略的最佳实践方案

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.3.3、 实现商铺和缓存与数据库双写一致

      核心思路如下:

      修改ShopController中的业务逻辑,满足下面的需求:

      根据id查询店铺时,如果缓存未命中,则查询数据库,将数据库结果写入缓存,并设置超时时间

      根据id修改店铺时,先修改数据库,再删除缓存

      修改重点代码1:修改ShopServiceImpl的queryById方法

      设置redis缓存时添加过期时间

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      修改重点代码2

      代码分析:通过之前的淘汰,我们确定了采用删除策略,来解决双写问题,当我们修改了数据之后,然后把缓存中的数据进行删除,查询时发现缓存中没有数据,则会从mysql中加载最新的数据,从而避免数据库和缓存不一致的问题

      @Transactional
      @Override
      public Result update(Shop shop) {
          Long id = shop.getId();
          if(id == null){
              return Result.fail("id不能为null");
          }
          this.updateById(shop);
          // 删除缓存
          stringRedisTemplate.delete(RedisConstants.CACHE_SHOP_KEY+id);
          return Result.ok();
      }
      

      2.4 缓存穿透

      2.4.1、问题的解决思路

      缓存穿透 :缓存穿透是指客户端请求的数据在缓存中和数据库中都不存在,这样缓存永远不会生效,这些请求都会打到数据库。

      常见的解决方案有两种:

      • 缓存空对象
        • 优点:实现简单,维护方便
        • 缺点:
          • 额外的内存消耗
          • 可能造成短期的不一致
      • 布隆过滤
        • 优点:内存占用较少,没有多余key
        • 缺点:
          • 实现复杂
          • 存在误判可能 (因为可能出现hash冲突)

      **缓存空对象思路分析:**当我们客户端访问不存在的数据时,先请求redis,但是此时redis中没有数据,此时会访问到数据库,但是数据库中也没有数据,这个数据穿透了缓存,直击数据库,我们都知道数据库能够承载的并发不如redis这么高,如果大量的请求同时过来访问这种不存在的数据,这些请求就都会访问到数据库,简单的解决方案就是哪怕这个数据在数据库中也不存在,我们也把这个数据存入到redis中去,这样,下次用户过来访问这个不存在的数据,那么在redis中也能找到这个数据就不会进入到数据库了

      布隆过滤: 布隆过滤器其实采用的是哈希思想来解决这个问题,通过一个庞大的二进制数组,走哈希思想去判断当前这个要查询的这个数据是否存在,如果布隆过滤器判断存在,则放行,这个请求会去访问redis,哪怕此时redis中的数据过期了,但是数据库中一定存在这个数据,在数据库中查询出来这个数据后,再将其放入到redis中,

      假设布隆过滤器判断这个数据不存在,则直接返回

      这种方式优点在于节约内存空间,存在误判,误判原因在于:布隆过滤器走的是哈希思想,只要哈希思想,就可能存在哈希冲突

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      代码修改:

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.4.2、编码解决商品查询的缓存穿透问题:

      核心思路如下:

      在原来的逻辑中,我们如果发现这个数据在mysql中不存在,直接就返回404了,这样是会存在缓存穿透问题的

      现在的逻辑中:如果这个数据不存在,我们不会返回404 ,还是会把这个数据写入到Redis中,并且将value设置为空,当再次发起查询时,我们如果发现命中之后,判断这个value是否是null,如果是null,则是之前写入的数据,证明是缓存穿透数据,如果不是,则直接返回数据。

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      小总结:

      缓存穿透产生的原因是什么?

      • 用户请求的数据在缓存中和数据库中都不存在,不断发起这样的请求,给数据库带来巨大压力

      缓存穿透的解决方案有哪些?

      • 缓存null值 (被动解决)
      • 布隆过滤 (被动解决)
      • 增强id的复杂度,避免被猜测id规律 (比如随机数,防止用户猜到…)
      • 做好数据的基础格式校验 (比如id的位数)
      • 加强用户权限校验
      • 做好热点参数的限流

      2.5 缓存雪崩问题及解决思路

      缓存雪崩是指在同一时段大量的缓存key同时失效或者Redis服务宕机,导致大量请求到达数据库,带来巨大压力。

      解决方案:

      • 给不同的Key的TTL添加随机值 (可以防止同一时间短缓存都失效~)
      • 利用Redis集群提高服务的可用性 (后面会讲)
      • 给缓存业务添加降级限流策略 (比如sentinel的一些策略)
      • 给业务添加多级缓存 (Nginx缓存、JVM缓存、MySQL缓存、Redis缓存 只要有一个命中就可~)

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.6 缓存击穿

      2.6.1、问题及解决思路

      缓存击穿问题也叫热点Key问题,就是一个被高并发访问并且缓存重建业务较复杂的key突然失效了,无数的请求访问会在瞬间给数据库带来巨大的冲击。

      常见的解决方案有两种:

      • 互斥锁
      • 逻辑过期

      逻辑分析:假设线程1在查询缓存之后,本来应该去查询数据库,然后把这个数据重新加载到缓存的,此时只要线程1走完这个逻辑,其他线程就都能从缓存中加载这些数据了,但是假设在线程1没有走完的时候,后续的线程2,线程3,线程4同时过来访问当前这个方法, 那么这些线程都不能从缓存中查询到数据,那么他们就会同一时刻来访问查询缓存,都没查到,接着同一时间去访问数据库,同时的去执行数据库代码,对数据库访问压力过大

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      解决方案一、使用锁来解决:

      因为锁能实现互斥性。假设线程过来,只能一个人一个人的来访问数据库,从而避免对于数据库访问压力过大,但这也会影响查询的性能,因为此时会让查询的性能从并行变成了串行,我们可以采用tryLock方法 + double check来解决这样的问题。

      假设现在线程1过来访问,他查询缓存没有命中,但是此时他获得到了锁的资源,那么线程1就会一个人去执行逻辑,假设现在线程2过来,线程2在执行过程中,并没有获得到锁,那么线程2就可以进行到休眠,直到线程1把锁释放后,线程2获得到锁,然后再来执行逻辑,此时就能够从缓存中拿到数据了。

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      解决方案二、逻辑过期方案

      方案分析:我们之所以会出现这个缓存击穿问题,主要原因是在于我们对key设置了过期时间,假设我们不设置过期时间,其实就不会有缓存击穿的问题,但是不设置过期时间,这样数据不就一直占用我们内存了吗,我们可以采用逻辑过期方案。

      我们把过期时间设置在 redis的value中,注意:这个过期时间并不会直接作用于redis,而是我们后续通过逻辑去处理。假设线程1去查询缓存,然后从value中判断出来当前的数据已经过期了,此时线程1去获得互斥锁,那么其他线程会进行阻塞,获得了锁的线程他会开启一个 线程去进行 以前的重构数据的逻辑,直到新开的线程完成这个逻辑后,才释放锁, 而线程1直接进行返回,假设现在线程3过来访问,由于线程线程2持有着锁,所以线程3无法获得锁,线程3也直接返回数据,只有等到新开的线程2把重建数据构建完后,其他线程才能走返回正确的数据。

      这种方案巧妙在于,异步的构建缓存,缺点在于在构建完缓存之前,返回的都是脏数据。

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      进行对比

      **互斥锁方案:**由于保证了互斥性,所以数据一致,且实现简单,因为仅仅只需要加一把锁而已,也没其他的事情需要操心,所以没有额外的内存消耗,缺点在于有锁就有死锁问题的发生,且只能串行执行性能肯定受到影响

      逻辑过期方案: 线程读取过程中不需要等待,性能好,有一个额外的线程持有锁去进行重构数据,但是在重构数据完成前,其他的线程只能返回之前的数据,且实现起来麻烦

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.6.2、利用互斥锁解决缓存击穿问题

      核心思路:相较于原来从缓存中查询不到数据后直接查询数据库而言,现在的方案是 进行查询之后,如果从缓存没有查询到数据,则进行互斥锁的获取,获取互斥锁后,判断是否获得到了锁,如果没有获得到,则休眠,过一会再进行尝试,直到获取到锁为止,才能进行查询

      如果获取到了锁的线程,再去进行查询,查询后将数据写入redis,再释放锁,返回数据,利用互斥锁就能保证只有一个线程去执行操作数据库的逻辑,防止缓存击穿

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      操作锁的代码:

      核心思路就是利用redis的setnx方法来表示获取锁,该方法含义是redis中如果没有这个key,则插入成功,返回1,在stringRedisTemplate中返回true, 如果有这个key则插入失败,则返回0,在stringRedisTemplate返回false,我们可以通过true,或者是false,来表示是否有线程成功插入key,成功插入的key的线程我们认为他就是获得到锁的线程。

      /**
       * 获取锁:使用setnx模拟互斥锁
       * 为了防止出现死锁,所以应该为其设置过期时间
       * @param key
       * @return
      */
      private boolean tryLock(String key){
          Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", RedisConstants.LOCK_SHOP_TTL, TimeUnit.SECONDS);
          return BooleanUtil.isTrue(flag);
      }
      
      /**
       * 释放锁
       * @param key
      */
      private void unlock(String key){
          stringRedisTemplate.delete(key);
      }
      

      操作代码:

      /**
       * 互斥锁解决缓存击穿
       * @param id
       * @return
      */
      private Shop queryWithMutex(Long id) {
          // 1.从查询Redis中是否有数据
          String shopJson = stringRedisTemplate.opsForValue().get(RedisConstants.CACHE_SHOP_KEY + id);
          // 2.判断是否存在
          if (StrUtil.isNotBlank(shopJson)) {
              // 存在则直接返回
              Shop shop = JSONUtil.toBean(shopJson, Shop.class);
              return shop;
          }
      
          // 3.判断命中的是否是空值
          if(shopJson != null){
              return null;
          }
      
          // 4.实现缓存重建
          String key = RedisConstants.LOCK_SHOP_KEY+id;
          Shop shop = null;
          try {
              // 4.1 获取互斥锁
              boolean isLock = tryLock(key);
              // 4.2判断是否获取成功
              if(!isLock){
                  // 4.3失败,则休眠并重试
                  Thread.sleep(50);
                  // 注意:获取锁的同时应该再次检测redis缓存是否存在,做DoubleCheck,如果存在则无需重建缓存
                  return queryWithMutex(id);
              }
              // 4.4成功,根据id查询数据库
              shop = getById(id);
      
              // 模拟重建时的延时
              Thread.sleep(200);
      
              // 5.不存在,返回错误
              if(shop==null){
                  // 将空值写入到redis
                  stringRedisTemplate.opsForValue().set(RedisConstants.CACHE_SHOP_KEY + id,"",RedisConstants.CACHE_NULL_TTL, TimeUnit.MINUTES);
                  return null;
              }
              // 6.存在就加入到Redis,并返回
              stringRedisTemplate.opsForValue().set(RedisConstants.CACHE_SHOP_KEY+id, JSONUtil.toJsonStr(shop),RedisConstants.CACHE_SHOP_TTL, TimeUnit.MINUTES);
          } catch (InterruptedException e) {
              e.printStackTrace();
          }finally {
              // 7.释放互斥锁
              unlock(key);
          }
          return shop;
      }
      

      测试:5s 1000次请求的高并发测试

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      观察控制台,可以发现只向数据库请求了一次~其他请求都打在了缓存上!!!

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      从而可以得出结论,使用互斥锁可以解决缓存击穿的问题~~~

      2.6.3、利用逻辑过期解决缓存击穿问题

      思路分析:当用户开始查询redis时,判断是否命中,如果没有命中则直接返回空数据,不查询数据库,而一旦命中后,将value取出,判断value中的过期时间是否满足,如果没有过期,则直接返回redis中的数据,如果过期,则在开启独立线程后直接返回之前的数据,独立线程去重构数据,重构完成后释放互斥锁。

      《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      封装数据:因为现在redis中存储的数据的value需要带上过期时间,此时要么你去修改原来的实体类,要么你新建一个实体类,我们采用第二个方案,这个方案,对原来代码没有侵入性。

      步骤一、 新建RedisData类

      @Data
      public class RedisData {
          private LocalDateTime expireTime;
          private Object data;
      }
      

      步骤二、 在ShopServiceImpl 新增此方法,利用单元测试进行缓存预热

      /**
       * 缓存重建(预热)
       * @param id
       * @param expireSeconds
       * @throws InterruptedException
      */
      @Override
      public void saveShop2Redis(Long id, Long expireSeconds) throws InterruptedException {
          // 1.查询店铺信息
          Shop shop = getById(id);
          Thread.sleep(200);
          // 2.封装逻辑过期时间
          RedisData redisData = new RedisData();
          redisData.setData(shop);
          redisData.setExpireTime(LocalDateTime.now().plusSeconds(expireSeconds));
          // 3.写入Redis
          stringRedisTemplate.opsForValue().set(RedisConstants.CACHE_SHOP_KEY + id, JSONUtil.toJsonStr(redisData));
      }
      
      @Test
      public void testSave2Redis() throws InterruptedException {
          shopService.saveShop2Redis(1L,30L);
      }
      

      步骤三:编写逻辑过期的核心代码

      private ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10);
      
      /**
       * 逻辑过期解决缓存击穿
       *
       * @param id
       * @return
      */
      private Shop queryWithLogicalExpire(Long id) {
          // 1.从Redis中查询商铺缓存
          String redisDataJson = stringRedisTemplate.opsForValue().get(RedisConstants.CACHE_SHOP_KEY + id);
          // 2.判断是否命中
          if (StrUtil.isBlank(redisDataJson)) {
              // 3.未命中,则返回空(因为预热过了,所以如果缓存中没有,则一定就是没有该店铺数据)
              // 为什么不用考虑缓存穿透的情况,因为所有的数据都放入到redis中预热了,一旦缓存查询出是null的,说明数据库没这个数据
              return null;
          }
          // 4.命中,需要先把json反序列化为对象
          RedisData redisData = JSONUtil.toBean(redisDataJson, RedisData.class);
          JSONObject shopJSONObj = (JSONObject) redisData.getData();
          Shop shop = JSONUtil.toBean(shopJSONObj, Shop.class);
          // 5.判断是否过期
          if (redisData.getExpireTime().isAfter(LocalDateTime.now())) {
              // 5.1未过期,则返回商铺信息
              return shop;
          }
          // 5.2已过期,需要缓存重建
          // 6.缓存重建
          // 6.1尝试获取互斥锁
          boolean isLock = tryLock(RedisConstants.LOCK_SHOP_KEY + id);
          // 6.2判断互斥锁是否获取成功
          if (isLock) {
              // 6.3获取成功,则开启独立线程
              CACHE_REBUILD_EXECUTOR.submit(() -> {
                  try {
                      // 6.4进行缓存重建
                      // 注意:这里写的是20S只是为了测试方便,实际项目中要用 30min
                      this.saveShop2Redis(id, 20L);
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  } finally {
                      // 6.5释放锁
                      unlock(RedisConstants.LOCK_SHOP_KEY + id);
                  }
              });
      
          }
          // 7.获取互斥锁失败,则直接返回过期的shop数据
          return shop;
      }
      

      步骤四:测试

      • 先执行单元测试方法预热缓存

      • 进行高并发请求 1s 100次

      • 观察请求的结果:缓存重建前返回的是旧数据,重建后得到的是新数据

        《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      • 观察控制台,也只有一条查询的SQL,说明互斥锁起到了作用!只缓存重建了一次~

        《Redis实战篇》二、商户查询缓存(缓存基本使用 | 缓存更新 | 缓存穿透 | 缓存雪崩 | 缓存击穿 | Redis缓存工具类)

      2.7、封装Redis工具类

      基于StringRedisTemplate封装一个缓存工具类,满足下列需求:

      • 方法1:将Java对象序列化为json并存储在string类型的key中,并且可以设置TTL过期时间
      • 方法2:根据指定的key查询缓存,并利用缓存空值来解决缓存穿透问题
      • 方法3:根据指定的key查询缓存,并利用互斥锁解决缓存击穿解决缓存击穿问题
      • 方法4:根据指定的key查询缓存,并反序列化为指定类型,需要利用逻辑过期解决缓存击穿问题

      代码如下:

      /**
       * @author lxy
       * @version 1.0
       * @Description Redis操作缓存的工具类
       * @date 2022/12/6 0:33
       */
      @Slf4j
      @Component
      public class CacheClient {
      
          private final StringRedisTemplate stringRedisTemplate;
      
          public CacheClient(StringRedisTemplate stringRedisTemplate) {
              this.stringRedisTemplate = stringRedisTemplate;
          }
      
          /**
           * 将Java对象序列化为json并存储在string类型的key中,并且可以设置TTL过期时间
           * @param key
           * @param value
           * @param time
           * @param unit
           */
          public void set(String key, Object value, Long time, TimeUnit unit){
              stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(value),time,unit);
          }
      
          /**
           *  逻辑过期解决缓存击穿问题中的缓存重建
           * @param key
           * @param value
           * @param time
           * @param unit
           */
          public void setWithLogicalExpire(String key,Object value,Long time,TimeUnit unit){
              // 设置逻辑过期
              RedisData redisData = new RedisData();
              redisData.setData(value);
              redisData.setExpireTime(LocalDateTime.now().plusSeconds(unit.toSeconds(time)));
              // 写入Redis
              stringRedisTemplate.opsForValue().set(key,JSONUtil.toJsonStr(redisData));
          }
      
          /**
           * 根据指定的key查询缓存,并利用缓存空值来解决缓存穿透问题
           * @param keyPrefix key前缀
           * @param id
           * @param type
           * @param dbFallback 降级的函数
           * @param time       时间
           * @param unit       单位
           * @param <R>
           * @param <ID>
           * @return
           */
          public <R,ID> R queryWithPassThrough(String keyPrefix, ID id, Class<R> type,
                                                Function <ID,R> dbFallback,Long time,TimeUnit unit) {
              String key = keyPrefix + id;
              // 1.从Redis中查询R数据
              String json = stringRedisTemplate.opsForValue().get(key);
              // 2.判断是否存在
              if (StrUtil.isNotBlank(json)) {
                  // 3.存在,则直接返回
                  return JSONUtil.toBean(json, type);
              }
      
              // 4.判断命中的是否是空值 (上面已经判断过不为空的情况了,下面只有 “” 和 null的两种情况,为null说明不存在,为“”说明空缓存)
              if (json != null) {
                  return null;
              }
      
              // 3.如果没有,就去查数据库
              R r = dbFallback.apply(id);
              // 4.如果没找到则返回错误信息
              if (r == null) {
                  stringRedisTemplate.opsForValue().set(key, "", RedisConstants.CACHE_NULL_TTL, TimeUnit.MINUTES);
                  return null;
              }
      
              // 5.如果查到了就加入到Redis,并返回
              this.set(key,r,time,unit);
              return r;
          }
      
          private ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10);
      
          /**
           * 根据指定的key查询缓存,并利用逻辑过期时间来解决缓存击穿问题
           *
           * @param id
           * @return
           */
          public <R,ID> R queryWithLogicalExpire(String keyPrefix,ID id,Class<R> type
          ,Function <ID,R> dbFallback,Long time,TimeUnit unit) {
              String key = keyPrefix + id;
              // 1.从Redis中查询商铺缓存
              String redisDataJson = stringRedisTemplate.opsForValue().get(key);
              // 2.判断是否命中
              if (StrUtil.isBlank(redisDataJson)) {
                  // 3.未命中,则返回空(因为预热过了,所以如果缓存中没有,则一定就是没有该店铺数据)
                  return null;
              }
              // 4.命中,需要先把json反序列化为对象
              RedisData redisData = JSONUtil.toBean(redisDataJson, RedisData.class);
              JSONObject jsonObj = (JSONObject) redisData.getData();
              R r = JSONUtil.toBean(jsonObj, type);
              // 5.判断是否过期
              if (redisData.getExpireTime().isAfter(LocalDateTime.now())) {
                  // 5.1未过期,则返回商铺信息
                  return r;
              }
              // 5.2已过期,需要缓存重建
              // 6.缓存重建
              // 6.1尝试获取互斥锁
              String lockKey = RedisConstants.LOCK_SHOP_KEY + id;
              boolean isLock = tryLock(lockKey);
              // 6.2判断互斥锁是否获取成功
              if (isLock) {
                  // 6.3获取成功,则开启独立线程
                  CACHE_REBUILD_EXECUTOR.submit(() -> {
                      try {
                          // 查询数据库
                          R newR = dbFallback.apply(id);
                          // 进行缓存重建
                          this.setWithLogicalExpire(key,newR,time,unit);
                      } catch (Exception e) {
                          e.printStackTrace();
                      } finally {
                          // 6.6释放锁
                          unlock(lockKey);
                      }
                  });
      
              }
              // 7.获取互斥锁失败,则直接返回过期的R数据
              return r;
          }
      
          /**
           * 根据指定的key查询缓存,并利用互斥锁解决缓存击穿解决缓存击穿问题
           *
           * @param id
           * @return
           */
          public <R,ID> R queryWithMutex(String keyPrefix,ID id,Class<R> type,
                                          Function <ID,R> dbFallback,Long time,TimeUnit unit) {
              String key = keyPrefix + id;
              // 1.从查询Redis中是否有数据
              String json = stringRedisTemplate.opsForValue().get(key);
              // 2.判断是否存在
              if (StrUtil.isNotBlank(json)) {
                  // 存在则直接返回
                  R r = JSONUtil.toBean(json, type);
                  return r;
              }
      
              // 3.判断命中的是否是 “” (缓存空值)
              if (json != null) {
                  return null;
              }
      
              // 4.实现缓存重建
              String lockKey = RedisConstants.LOCK_SHOP_KEY + id;
              R r = null;
              try {
                  // 4.1 获取互斥锁
                  boolean isLock = tryLock(lockKey);
                  // 4.2判断是否获取成功
                  if (!isLock) {
                      // 4.3失败,则休眠并重试
                      Thread.sleep(50);
                      // 注意:获取锁的同时应该再次检测redis缓存是否存在,做DoubleCheck,如果存在则无需重建缓存
                      return queryWithMutex(keyPrefix,id,type,dbFallback,time,unit);
                  }
                  // 4.4成功,根据id查询数据库
                  r = dbFallback.apply(id);
      
                  // 模拟重建时的延时
                  Thread.sleep(200);
      
                  // 5.不存在,返回错误
                  if (r == null) {
                      // 将空值写入到redis
                      stringRedisTemplate.opsForValue().set(key, "", RedisConstants.CACHE_NULL_TTL, TimeUnit.MINUTES);
                      return null;
                  }
                  // 6.存在就加入到Redis,并返回
                  stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(r),time, unit);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              } finally {
                  // 7.释放互斥锁
                  unlock(lockKey);
              }
              return r;
          }
      
          /**
           * 获取锁:使用setnx模拟互斥锁
           * 为了防止出现死锁,所以应该为其设置过期时间
           *
           * @param key
           * @return
           */
          private boolean tryLock(String key) {
              Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", RedisConstants.LOCK_SHOP_TTL, TimeUnit.SECONDS);
              return BooleanUtil.isTrue(flag);
          }
      
          /**
           * 释放锁
           *
           * @param key
           */
          private void unlock(String key) {
              stringRedisTemplate.delete(key);
          }
      }
      

      可能有人好奇为什么工具类的方法不用static修饰?

      ioc是通过new的方式创建bean,new出来的对象是在堆里面, static修饰的东西是优先于对象存在,如果我们用了static,那么方法里面涉及到的方法调也得是Static,但是我们因为用的是ioc容器里面的bean,所以不能这样搞!

      在ShopServiceImpl 中测试工具类

      @Override
      public Result queryById(Long id) {
          // 缓存穿透
          // Shop shop = cacheClient.queryWithPassThrough(RedisConstants.CACHE_SHOP_KEY, id, Shop.class,
          //         this::getById, RedisConstants.CACHE_SHOP_TTL, TimeUnit.MINUTES);
      
          // 互斥锁解决缓存击穿
          // Shop shop = cacheClient.queryWithMutex(RedisConstants.CACHE_SHOP_KEY,id,Shop.class,this::getById,RedisConstants.CACHE_SHOP_TTL,TimeUnit.MINUTES);
      
          // 逻辑过期解决缓存击穿
          Shop shop = cacheClient.queryWithLogicalExpire(RedisConstants.CACHE_SHOP_KEY,id,Shop.class,this::getById,RedisConstants.CACHE_SHOP_TTL,TimeUnit.MINUTES);
      
          if (shop == null) {
              return Result.fail("店铺不存在!");
          }
          // 返回
          return Result.ok(shop);
      }
      版权声明:本文内容来自第三方投稿或授权转载,原文地址:https://blog.csdn.net/LXYDSF/article/details/128180192,作者:爱编程的大李子,版权归原作者所有。本网站转在其作品的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如因作品内容、版权等问题需要同本网站联系,请发邮件至ctyunbbs@chinatelecom.cn沟通。

      上一篇:DVWA之Stored XSS(存储型XSS)

      下一篇:你觉得redis有什么缺点,给你改进的话你会怎么改进?

      相关文章

      2025-05-14 10:03:13

      【Mybatis】-防止SQL注入

      【Mybatis】-防止SQL注入

      2025-05-14 10:03:13
      SQL , 执行 , 日志 , 注入 , 缓存 , 编译 , 语句
      2025-05-14 10:02:58

      java休眠到指定时间怎么写

      java休眠到指定时间怎么写

      2025-05-14 10:02:58
      java , sleep , Thread , util , 方法
      2025-05-14 10:02:58

      java项目多端数据同步解决方案

      多端数据同步是指在多个设备(例如桌面应用、移动应用、Web应用)之间保持数据的一致性。

      2025-05-14 10:02:58
      java , Spring , WebSocket , 同步 , 数据 , 版本号
      2025-05-14 10:02:48

      互斥锁解决redis缓存击穿

      在高并发系统中,Redis 缓存是一种常见的性能优化方式。然而,缓存击穿问题也伴随着高并发访问而来。

      2025-05-14 10:02:48
      Redis , 互斥 , 数据库 , 线程 , 缓存 , 请求
      2025-05-13 09:50:28

      分隔链表-146. LRU 缓存

      给你一个链表的头节点 head 和一个特定值 x ,请你对链表进行分隔,使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。

      2025-05-13 09:50:28
      int , key , LinkedHashMap , 缓存 , 节点 , 链表
      2025-05-13 09:49:12

      Java学习(动态代理的思想详细分析与案例准备)(1)

      Java学习(动态代理的思想详细分析与案例准备)(1)

      2025-05-13 09:49:12
      java , 代理 , 代码 , 对象 , 接口 , 方法 , 需要
      2025-05-12 08:43:47

      LRU 缓存

      请你设计并实现一个满足  LRU (最近最少使用) 缓存 约束的数据结构。

      2025-05-12 08:43:47
      int , key , lt , 关键字 , 缓存
      2025-05-09 08:20:32

      基于IDEA的Maven简单工程创建及结构分析

      通过一个 mvn 命令直接让我们创建一个 Maven 的脚手架。

      2025-05-09 08:20:32
      java , Maven , xml , 创建 , 文件 , 文件夹 , 项目
      2025-05-08 09:04:25

      DS初阶:时间复杂度和空间复杂度

      算法(Algorithm):就是定义良好的计算过程,他取一个或一组的值为输入,并产生出一个或一组值作为输出。简单来说算法就是一系列的计算步骤,用来将输入数据转化成输出结果。                                                      

      2025-05-08 09:04:25
      CPU , 复杂度 , 数据 , 时间 , 空间 , 算法 , 缓存
      2025-05-08 09:03:57

      前K个高频元素java

      给定一个非空的整数数组,返回其中出现频率前 前K个高频元素java 高的元素。

      2025-05-08 09:03:57
      java , 元素 , 样例 , 给定
      查看更多
      推荐标签

      作者介绍

      天翼云小翼
      天翼云用户

      文章

      33561

      阅读量

      5238101

      查看更多

      最新文章

      【Mybatis】-防止SQL注入

      2025-05-14 10:03:13

      分隔链表-146. LRU 缓存

      2025-05-13 09:50:28

      LRU 缓存

      2025-05-12 08:43:47

      DS初阶:时间复杂度和空间复杂度

      2025-05-08 09:04:25

      springboot系列教程(十三):基于Cache注解模式,管理Redis缓存

      2025-05-07 09:08:42

      【Linux 从基础到进阶】Redis缓存服务安装与调优

      2025-05-06 09:18:38

      查看更多

      热门文章

      Redis持久化存储策略(RDB、AOF)

      2023-05-24 08:11:04

      查看Redis的默认设置的过期策略和内存淘汰机制

      2022-12-28 07:22:30

      leetcode数据结构-LRU

      2023-03-02 10:21:35

      精华推荐 | 【深入浅出RocketMQ原理及实战】「底层原理挖掘系列」透彻剖析贯穿RocketMQ的存储系统的实现原理和持久化机制

      2023-02-24 10:12:47

      elasticsearch预加载数据到文件系统缓存

      2024-09-25 10:13:57

      StringBuffer.append(str)之OutOfMemoryError打破砂锅问到底

      2022-12-28 07:22:30

      查看更多

      热门标签

      存储 缓存 内存 数据库 数据 redis mysql 服务器 数据恢复 Redis linux java 链表 MySQL sql
      查看更多

      相关产品

      弹性云主机

      随时自助获取、弹性伸缩的云服务器资源

      天翼云电脑(公众版)

      便捷、安全、高效的云电脑服务

      对象存储

      高品质、低成本的云上存储服务

      云硬盘

      为云上计算资源提供持久性块存储

      查看更多

      随机文章

      redis 如何保证缓存和数据库一致性?

      Redis中的数据类型

      浅聊缓存函数

      MySQL查询缓存

      【实战问题】-- 缓存穿透,缓存击穿和缓存雪崩的区别以及解决方案

      python中functools.cache用法详解及缓存策略问题

      • 7*24小时售后
      • 无忧退款
      • 免费备案
      • 专家服务
      售前咨询热线
      400-810-9889转1
      关注天翼云
      • 旗舰店
      • 天翼云APP
      • 天翼云微信公众号
      服务与支持
      • 备案中心
      • 售前咨询
      • 智能客服
      • 自助服务
      • 工单管理
      • 客户公告
      • 涉诈举报
      账户管理
      • 管理中心
      • 订单管理
      • 余额管理
      • 发票管理
      • 充值汇款
      • 续费管理
      快速入口
      • 天翼云旗舰店
      • 文档中心
      • 最新活动
      • 免费试用
      • 信任中心
      • 天翼云学堂
      云网生态
      • 甄选商城
      • 渠道合作
      • 云市场合作
      了解天翼云
      • 关于天翼云
      • 天翼云APP
      • 服务案例
      • 新闻资讯
      • 联系我们
      热门产品
      • 云电脑
      • 弹性云主机
      • 云电脑政企版
      • 天翼云手机
      • 云数据库
      • 对象存储
      • 云硬盘
      • Web应用防火墙
      • 服务器安全卫士
      • CDN加速
      热门推荐
      • 云服务备份
      • 边缘安全加速平台
      • 全站加速
      • 安全加速
      • 云服务器
      • 云主机
      • 智能边缘云
      • 应用编排服务
      • 微服务引擎
      • 共享流量包
      更多推荐
      • web应用防火墙
      • 密钥管理
      • 等保咨询
      • 安全专区
      • 应用运维管理
      • 云日志服务
      • 文档数据库服务
      • 云搜索服务
      • 数据湖探索
      • 数据仓库服务
      友情链接
      • 中国电信集团
      • 189邮箱
      • 天翼企业云盘
      • 天翼云盘
      ©2025 天翼云科技有限公司版权所有 增值电信业务经营许可证A2.B1.B2-20090001
      公司地址:北京市东城区青龙胡同甲1号、3号2幢2层205-32室
      • 用户协议
      • 隐私政策
      • 个人信息保护
      • 法律声明
      备案 京公网安备11010802043424号 京ICP备 2021034386号