总结一下Redis的缓存雪崩、缓存击穿、缓存穿透

news2025/1/14 0:53:57

缓存是提高系统性能的一种常见手段,其中Redis是一种常用的高性能缓存数据库。但是在使用缓存时,可能会遇到一些问题,比如缓存击穿、缓存穿透、缓存雪崩等问题,本文将介绍这些问题的概念、原因以及解决方案。

缓存击穿

缓存击穿指的是在高并发情况下,一个缓存的key在缓存中不存在,导致每次请求都要访问数据库,从而导致数据库压力过大,甚至崩溃。这种情况通常发生在一些热点数据上,比如用户登录信息等。

原因

缓存击穿的原因是因为在某些热点数据的key失效或者被删除时,大量的并发请求同时访问这个key,导致缓存中不存在这个key的数据,从而每个请求都需要去访问数据库获取数据,造成数据库压力过大。

解决方案

  • 1.设置热点数据永不过期

在缓存中设置热点数据永不过期可以有效地避免缓存击穿问题。但是这种方式会导致缓存中存在很多过期但是占用内存的数据,因此需要在设置缓存数据时进行权衡。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
    value = db.get(key);
    if (value != null) {
        redis.set(key, value);
        redis.persist(key); //设置key永不过期
    }
}
复制代码
  • 2.设置热点数据短期过期

为了避免缓存中过多占用内存的数据,可以将热点数据设置一个相对较短的过期时间,比如1分钟,这样可以避免过期数据占用过多内存。当热点数据过期后,可以在后台异步更新缓存数据。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
    //添加分布式锁,避免缓存穿透
    if(redis.setNx("lock_"+key,"value")){
        value = db.get(key);
        if (value != null) {
            redis.set(key, value);
            redis.expire(key,60); //设置key过期时间为1分钟
        }
        redis.del("lock_"+key);
    }else {
        Thread.sleep(50);
        return queryDataFromCache(key);
    }
}
复制代码

缓存穿透

缓存穿透指的是当大量的并发请求同时查询一个不存在的key时,由于缓存中没有对应的数据,所以每个请求都会去访问数据库,导致数据库压力过大。

原因

缓存穿透的原因是由于黑客攻击或者恶意请求,可能会对某些不存在的数据进行大量的请求,从而导致缓存穿透问题。

解决方案

  • 1.对查询结果为空的key设置空值

当缓存查询的结果为空时,可以将结果设置为空值写入缓存,这样下次查询相同的key时,可以直接从缓存中获取结果,避免了查询数据库的开销。

String key = "not_exist_data";
String value = redis.get(key);
if (value == null) {
    //添加分布式锁,避免缓存穿透
    if(redis.setNx("lock_"+key,"value")){
        value = db.get(key);
        if (value != null) {
            redis.set(key, value);
        }else {
            redis.set(key, ""); //设置空值
            redis.expire(key, 60); //设置过期时间为1分钟
        }
        redis.del("lock_"+key);
    }else {
        Thread.sleep(50);
        return queryDataFromCache(key);
    }
}
复制代码
  • 2.BloomFilter过滤非法请求

使用BloomFilter可以对请求参数进行过滤,将非法请求拦截在系统外部,从而避免了对系统的压力。

BloomFilter filter = new BloomFilter(10000, 0.001); //设置布隆过滤器
String key = "not_exist_data";
if(filter.mightContain(key)){
    return null;
}
String value = redis.get(key);
if (value == null) {
    //添加分布式锁,避免缓存穿透
    if(redis.setNx("lock_"+key,"value")){
        value = db.get(key);
        if (value != null) {
            redis.set(key, value);
        }else {
            filter.put(key); //将非法key加入过滤器
        }
        redis.del("lock_"+key);
    }else {
        Thread.sleep(50);
        return queryDataFromCache(key);
    }
}
复制代码

缓存雪崩

缓存雪崩指的是在缓存中存在大量的key过期时间相同或者失效的情况下,当这些key同时失效时,大量的并发请求都会涌入数据库,导致数据库压力过大,甚至崩溃。

原因

缓存雪崩的原因是因为在缓存中存在大量的key同时过期,导致大量的并发请求同时涌入数据库。

解决方案

  • 1.缓存数据随机过期时间

为了避免缓存中大量key同时过期,可以设置每个缓存数据的过期时间不同,比如可以在原有过期时间的基础上添加一个随机时间,这样可以避免大量key同时过期的情况。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
    //添加分布式锁,避免缓存穿透
    if(redis.setNx("lock_"+key,"value")){
        value = db.get(key);
        if (value != null) {
            //设置随机过期时间,避免缓存雪崩
            Random random = new Random();
            int expireTime = random.nextInt(1800) + 1800; //过期时间在30~60分钟之间
            redis.set(key, value);
            redis.expire(key, expireTime);
        }
        redis.del("lock_"+key);
    }else {
        Thread.sleep(50);
        return queryDataFromCache(key);
    }
}
复制代码
  • 2.缓存数据预加载

为了避免在缓存中大量的key失效,可以在缓存数据过期之前,提前将缓存数据刷新到缓存中,保证数据的可用性。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
    //添加分布式锁,避免缓存穿透
    if(redis.setNx("lock_"+key,"value")){
        value = db.get(key);
        if (value != null) {
            redis.set(key, value);
            redis.expire(key, 1800); //设置过期时间为30分钟
        }
        redis.del("lock_"+key);
    }else {
        Thread.sleep(50);
        return queryDataFromCache(key);
    }
}else {
    //判断缓存是否需要刷新
    if(redis.ttl(key) < 300){
        new Thread(() -> {
            String newValue = db.get(key);
            if (newValue != null) {
                redis.set(key, newValue);
                redis.expire(key, 1800); //设置过期时间为30分钟
            }
        }).start();
    }
}
复制代码
  • 3.限流降级

当缓存雪崩问题出现时,可以通过限流降级的方式来减少对数据库的请求,从而保证系统的可用性。可以通过配置Hystrix等限流降级框架来实现。

String key = "hot_data";
String value = redis.get(key);
if (value == null) {
    //使用Hystrix进行限流降级
    value = HystrixCommand.execute(() -> {
        String data = db.get(key);
        redis.set(key, data);
        redis.expire(key, 1800); //设置过期时间为30分钟
        return data;
    }, () -> {
        return "系统繁忙,请稍后重试!";
    });
}
复制代码

总结

Redis的使用,可以有效地提高系统的性能和可用性。但是在使用过程中,需要注意缓存击穿、缓存穿透和缓存雪崩等问题,采用适当的解决方案来避免这些问题的发生,从而保证系统的稳定性和可靠性。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/428812.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Jenkins——用户管理、授权策略配置以及Jenkins安全配置管理

这里写目录标题一、Jenkins用户管理1、进入系统管理界面2、创建用户3、编辑用户信息4、编辑用户信息5、删除用户信息二、Jenkins授权策略配置1、授权策略插件2、安装插件:Role-based Authorization Strategy3、管理角色a、全局角色 Global rolesb、项目角色 Item rolesc、节点角…

CVPR | 达摩院开源自监督学习框架CoKe, 单机8卡可训练

团队模型、论文、博文、直播合集&#xff0c;点击此处浏览 一、论文 论文链接: Unsupervised Visual Representation Learning by Online Constrained K-Means 代码链接&#xff1a;https://github.com/idstcv/CoKe 二、背景 虽然基于instance的自监督学习方法在ImageNet上…

注册claude AI账号 slack工作区账号

Claude 是建立在 slack工作区的一个AI人工助手&#xff0c;更像是将chatgpt集成到了会议模式&#xff0c;一个账号实际上拥有了你的会议室和你的AI助手&#xff0c;你可以让你的朋友和同事进入你的房间体验。 Claude是不是openai的产物&#xff1f;目前还不知道&#xff0c;不…

RS232/RS485/RS422 接线说明

RS232/RS485/RS422 接线 文章目录RS232/RS485/RS422 接线RS232RS485422RS232 设备A 和 设备B 没有硬件流控 设备A设备BTXDRXDRXDTXDGNDGND 设备A 和 设备B 使用RTS/CTS 做硬件流控 一般使用RTS/CTS都会由232芯片自动控制&#xff0c; 比如XR21V1414, 需要设备gpio mode 为00…

Docker安装+利用docker安装MySQL(保姆级教程)

前言&#xff1a;本文基于docker安装MySQL&#xff0c;如果还没安装docker的童鞋&#xff0c;点击这里&#xff08;超大声&#xff09;安装卸载老版本docker&#xff08;超详细&#xff09; 一、在docker仓库搜你想要的镜像版本 docker镜像仓库 打开官网&#xff0c;在搜索框…

Docker 镜像制作 服务编排 私有仓库

DockerNginx部署Redis部署Dockerfile镜像制作容器转为镜像dockerfileDocker服务编排Docker ComposeDocker Compose安装使用docker compose编排nginxspringboot项目Docker私有仓库私有仓库搭建将镜像上传至私有仓库Nginx部署 案例&#xff1a;需求 在Docker容器中部署Nginx&…

【ARMv8 异常模型入门及渐进2 - 系统寄存器访问方法:op1,CRn,CRm,op2】

文章目录1.1 ARMv8 系统寄存器访问概要1.1.1 系统寄存器访问级别1.1.2 ARMv8 系统寄存器编码1.1 ARMv8 系统寄存器访问概要 ARMv8中&#xff0c;取消了协处理器&#xff0c;之前协处理器实现的功能&#xff0c;全部由系统寄存器来是实现。对于系统寄存器的访问&#xff0c;使用…

javaweb在校大学生贷款管理系统ns08a9

1系统主要实现&#xff1a;学生注册、填写详细资料、申请贷款、学校审核、银行审核、贷后管理等功能&#xff0c; (1) 学生注册&#xff1a;学生通过注册用户&#xff0c;提交自己的详细个人资料&#xff0c;考虑现实应用中的安全性&#xff0c;资料提交后不可修改&#xff1b;…

Springboot + MySQL+ JPA Ⅲ delete方法详解

一、deleteById(Id id) 和 delete(T entity) 为什么要把这两个方法放在一起呢&#xff1f;我们先看源码再说 deleteById源码&#xff08;通过id进行删除&#xff09; Transactional Override public void deleteById(ID id) {Assert.notNull(id, ID_MUST_NOT_BE_NULL);delet…

@爱打游戏的你,当游戏测试是什么感觉?

爱打游戏的你&#xff0c;当游戏测试是一种什么感觉&#xff1f; 去年《宝可梦朱紫》大火的那段时间&#xff0c;想必各位爱好游戏的友友们都刷到过这样的图吧&#xff1a;&#xff08;量子纠缠&#xff09;&#xff08;天怎么黑了&#xff09;&#xff08;弹簧巨怪&#xff09…

手敲Mybatis(九)-结果集处理器

1.前言-背景介绍 上节我们处理了参数处理器&#xff0c;本节我们处理结果集处理器&#xff0c;之前我们写了一个DefaultResultSetHandler&#xff0c;我们把返回结果获取对象&#xff0c;填充值什么的写到了一起&#xff0c;流程没有进行解耦&#xff0c;并且只接收了Object的…

不通过鲁大师进行硬件检测

文章目录简介dxdiag系统信息设备管理器任务管理器PowerShelldxdiag系统信息设备管理器任务管理器PowerShellTODO&#xff1a;CPU-ZTODO&#xff1a;Everest参考文献简介 不少二手电脑重装系统后发现是假配置&#xff08;特别是XP系统&#xff09;&#xff0c;可以使用系统自带…

基于Vue2实现滚动过程中数据懒加载

以下为实现滚动懒加载的过程&#xff1a; 1、在data对象中定义了items数组&#xff0c;用于存放已加载的item&#xff0c;loading状态&#xff0c;当前页数page&#xff0c;每页数量pageSize&#xff0c;以及距离底部的阈值threshold。 2、在mounted钩子函数中&#xff0c;首次…

Adaptive AUTOSAR——State Management(VRTE 3.0 R21-11)

状态管理是自适应平台服务中的一个功能集群。 在自适应平台中&#xff0c;状态决定了一组活动的自适应应用程序。 特定于项目的应用程序&#xff0c;即状态管理器&#xff0c;决定何时请求状态更改&#xff0c;从而更改当前活动的应用程序集。状态管理器是特定于项目的&#…

【Golang】三分钟让你快速了解Go语言为什么我们需要Go语言?

博主简介&#xff1a;努力学习的大一在校计算机专业学生&#xff0c;热爱学习和创作。目前在学习和分享&#xff1a;数据结构、Go&#xff0c;Java等相关知识。博主主页&#xff1a; 是瑶瑶子啦所属专栏: Go语言核心编程近期目标&#xff1a;写好专栏的每一篇文章 目录一、Go语…

Pytorch基础 - 3. torch.utils.tensorboard

目录 1. 简介 2. 基本步骤 3. 示例1 - 可视化单条曲线 4. 示例2 - 可视化多条曲线 5. 示例3 - 可视化网络结构 1. 简介 Tensorboard是Tensorflow的可视化工具&#xff0c;常用来可视化网络的损失函数&#xff0c;网络结构&#xff0c;图像等。后来将Tensorboard集成到了P…

wps如何修改已经存在的目录标题内容?

如需了解更多办公应用的相关知识&#xff0c;可进入到赛效官网查看应用资讯或者应用问答栏目。 在WPS文档中&#xff0c;为方便大家查看文档中的内容&#xff0c;可以给文档内容添加目录&#xff0c;很多人由于对文档功能不太熟悉&#xff0c;所以当目录生成以后&#xff0c;想…

高效又稳定的ChatGPT大模型训练技巧总结,让训练事半功倍!

文&#xff5c;python前言近期&#xff0c;ChatGPT成为了全网热议的话题。ChatGPT是一种基于大规模语言模型技术&#xff08;LLM&#xff0c; large language model&#xff09;实现的人机对话工具。现在主流的大规模语言模型都采用Transformer网络&#xff0c;通过极大规模的数…

Day941.仓库版本管理 -系统重构实战

仓库&版本管理 Hi&#xff0c;我是阿昌&#xff0c;今天学习记录的是关于仓库&版本管理的内容。 当代码以及团队达到一定的规模以后&#xff0c;这会给项目仓库和版本的管理带来诸多问题。 一张常见问题的表格&#xff0c;不妨“对号入座”&#xff0c;看看项目有没…

【常见CSS扫盲雪碧图】从源码细看CSS雪碧图原理及实现,千字详解【附源码demo下载】

【写在前面】其实估计很多人都听过雪碧图&#xff0c;或者是CSS-Sprite&#xff0c;在很多门户网站就会经常有用到的&#xff0c;之所有引出雪碧图这个概念还得从前端加载多个图片时候页面闪了一下说起&#xff0c;这样给人的视觉效果体验很差&#xff0c;也就借此机会和大家说…