Redis学习——高级篇⑦

news2025/2/2 14:47:36

Redis学习——高级篇⑦

    • = = = = = = = Redis7之缓存预热 + 缓存雪崩 + 缓存击穿 + 缓存穿透(八) = = = = = = = =
    • 8.1 缓存预热
      • 8.1.1 是什么
      • 8.1.2 解决
    • 8.2 缓存雪崩
      • 8.2.1 是什么
      • 8.2.2 发生
      • 8.2.3 预防 + 解决
    • 8.3 缓存穿透
      • 8.3.1 是什么
      • 8.3.2 解决
        • 1 空对象缓存或者缺省值
        • 2 Google布隆过滤器Guava
    • 8.4 缓存击穿
      • 8.4.1 是什么
      • 8.4.2 解决
      • 8.4.3 案例编码(防止缓存击穿)
    • 8.5 总结

在这里插入图片描述

在这里插入图片描述

= = = = = = = Redis7之缓存预热 + 缓存雪崩 + 缓存击穿 + 缓存穿透(八) = = = = = = = =

在这里插入图片描述

8.1 缓存预热

8.1.1 是什么

在这里插入图片描述

缓存预热就是系统上线后,提前将相关的缓存数据直接加载到缓存系统。避免在用户请求的时候,先查询数据库,然后再将数据缓存的问题!用户直接查询事先被预热的缓存数据!

mysq|有100条新记录,redis无

  1. 比较懒,什么都不做,之对mysq|做了数据新增,利用redis的回写机制,让它逐步实现100条新增记录的同步最好提前晚上部署发布版本的时候,由自己人提前做一次,让redis同步了,不要把这个问题留给客户
  2. 通过中间件或者程序自行完成。

在这里插入图片描述

8.1.2 解决

使用 @PostConstruct 初始化白名单数据
在这里插入图片描述

8.2 缓存雪崩

8.2.1 是什么

缓存雪崩就是瞬间过期数据量太大,导致对数据库服务器造成压力。

8.2.2 发生

  • redis 主机挂了, Redis全盘崩溃,偏硬件运维
  • redis 中有大量key 同时过期大面积失效,偏软件开发

8.2.3 预防 + 解决

  • redis 中 key 设置为永不过期 or 过期时间错开
  • redis 缓存集群实现高可用
    • 主从 + 哨兵
    • Redis 集群
    • 开启Redis 持久化机制 aof / rdb,尽快恢复缓存集群
  • 多缓存结合预防雪崩
    • ehcache 本地缓存 + redis缓存

    • 在这里插入图片描述

在这里插入图片描述

8.3 缓存穿透

8.3.1 是什么

缓存穿透 就是请求去查询一条数据,先查redis,redis里面没有,再查mysql,mysql里面无,都查询不到该条记录,但是请求每次都会打到数据库上面去,导致后台数据库压力暴增

这个redis变成了一个摆设。 。。。。。简单说就是:本来无一物, 两库都没有。既不在Redis缓存库,也不在mysq|,数据库存在被多次暴击风险

8.3.2 解决

缓存穿透最怕恶意攻击,解决方案有2个:空对象缓存bloomfilter过滤器

在这里插入图片描述

1 空对象缓存或者缺省值

如果发生缓存穿透,可以针对要查询的数据,在Redis里存一个和业务部门商量后确定的缺省值(比如 零、负数、defaultNull等)

public Customer findCustomerById(Integer customerId) {
    Customer customer = null;
    // 缓存redis的key名称
    String key = CACHE_KEY_CUSTOMER + customerId;
    // 1.去redis上查询
    customer = (Customer) redisTemplate.opsForValue().get(key);

    // 2. 如果redis有,直接返回  如果redis没有,在mysql上查询
    if (customer == null) {
        // 3.对于高QPS的优化,进来就先加锁,保证一个请求操作,让外面的redis等待一下,避免击穿mysql(大公司的操作 )
        synchronized (CustomerService.class) {
            // 3.1 第二次查询redis,加锁后
            customer = (Customer) redisTemplate.opsForValue().get(key);
            // 4.再去查询我们的mysql
            customer = customerMapper.selectByPrimaryKey(customerId);

            // 5.mysql有,redis无
            if (customer != null) {
                // 6.把mysql查询到的数据会写到到redis, 保持双写一致性  7天过期
                redisTemplate.opsForValue().set(key, customer, 7L, TimeUnit.DAYS);
            }else {
                // defaultNull 规定为redis查询为空、MySQL查询也没有,缓存一个defaultNull标识为空,以防缓存穿透
                redisTemplate.opsForValue().set(key, "defaultNull", 7L, TimeUnit.DAYS);
            }
        }
    }
    return customer;
}
2 Google布隆过滤器Guava

案例:白名单过滤器

  • POM
<!--guava Google 开源的 Guava 中自带的布隆过滤器-->
<dependency>
   <groupId>com.google.guava</groupId>
   <artifactId>guava</artifactId>
   <version>23.0</version>
</dependency>
  • 业务类

  • GUavaBloomFilterController

import com.xfcy.service.GuavaBloomFilterService;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;

import javax.annotation.Resource;

@Api(tags = "gogle工具Guava处理布隆过滤器")
@RestController
@Slf4j
public class GuavaBloomFilterController {

    @Resource
    private GuavaBloomFilterService guavaBloomFilterService;

    @ApiOperation("guava布隆过滤器插入100万样本数据,额外10w(110w)测试是否存在")
    @RequestMapping(value = "/guavafilter", method = RequestMethod.GET)
    public void guavaBloomFilter() {
        guavaBloomFilterService.guavaBloomFilter();
    }
}
  • GUavaBloomFilterService
import com.google.common.hash.BloomFilter;
import com.google.common.hash.Funnels;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;

import java.util.ArrayList;

@Slf4j
@Service
public class GuavaBloomFilterService {
    // 1.定义一个常量
    public static final int _1W = 10000;
    // 2.定义我们guava布隆过滤器,初始容量
    public static final int SIZE = 100 * _1W;
    // 3.误判率,它越小误判的个数也越少(思考:是否可以无限小? 没有误判岂不是更好)
    public static double fpp = 0.01;  // 这个数越小所用的hash函数越多,bitmap占用的位越多  默认的就是0.03,5个hash函数   0.01,7个函数
    // 4.创建guava布隆过滤器
    private static BloomFilter<Integer> bloomFilter = BloomFilter.create(Funnels.integerFunnel(), SIZE, fpp);

    public void guavaBloomFilter() {
        // 1.先让bloomFilter加入100w白名单数据
        for (int i = 0; i < SIZE; i++) {
            bloomFilter.put(i);
        }
        // 2.故意取10w个不在合法范围内的数据,来进行误判率的演示
        ArrayList<Integer> list = new ArrayList<>(10 * _1W);

        // 3.验证
        for (int i = SIZE + 1; i < SIZE + (10 * _1W); i++){
            if (bloomFilter.mightContain(i)){
                log.info("被误判了:{}", i);
                list.add(i);
            }
        }
        log.info("误判总数量:{}", list.size());
    }
}

在这里插入图片描述

在这里插入图片描述

8.4 缓存击穿

在这里插入图片描述

8.4.1 是什么

缓存击穿就是大量请求同时查询一个key时,此时这个key正好失效了,就会导致大量的请求都打到数据库上面去,也就是热点key突然都失效了,MySQL承受高并发量

8.4.2 解决

在这里插入图片描述

  1. 差异失效时间,对于访问频繁的热点key,干脆就不设置过期时间

  2. 互斥更新,采用双检加锁
    在这里插入图片描述

8.4.3 案例编码(防止缓存击穿)

对于分页显示数据,在高并发下,绝对不能使用mysql,可以用redis的list结构

差异失效时间 用在双缓存架构

在这里插入图片描述

步骤说明
1100%高并发,绝对不可以用mysql实现
2先把mysql里面参加活动的数据抽取进redis,一般采用定时器扫描来决定上线活动还是下线取消。
3支持分页功能,一页20条记录
X请大家思考,redis里面什么样子的数据类型支持上述功能?

Product类

import io.swagger.annotations.ApiModel;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;


@ApiModel(value = "聚划算活动product信息")
@AllArgsConstructor
@NoArgsConstructor
@Data
public class Product {
    // 产品id
    private Long id;
    // 产品名称
    private String name;
    // 产品价格
    private Integer price;
    // 产品详情
    private String detail;
}

JHSTaskService(采用定时器将参加活动的商品加入redis)

import com.xfcy.entities.Product;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;

import javax.annotation.PostConstruct;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.concurrent.TimeUnit;

@Service
@Slf4j
public class JHSTaskService {

    public static final String JHS_KEY = "jhs";
    public static final String JHS_KEY_A = "jhs:a";
    public static final String JHS_KEY_B = "jhs:b";

    @Autowired
    private RedisTemplate redisTemplate;

    /**
     * 模拟从数据库读取20件特价商品
     * @return
     */
    private List<Product> getProductsFromMysql() {
        List<Product> list = new ArrayList<>();
        for (int i = 0; i <= 20; i++) {
            Random random = new Random();
            int id = random.nextInt(10000);
            Product product = new Product((long) id, "product" + i, i, "detail");
            list.add(product);
        }
        return list;
    }

    //@PostConstruct    // 测试单缓存
    public void initJHS(){
        log.info("启动定时器 天猫聚划算模拟开始 ===============");

        // 1.用线程模拟定时任务,后台任务定时将mysql里面的特价商品刷新的redis
        new Thread(() -> {
            while (true){
                // 2.模拟从mysql查到数据,加到redis并返回给页面
                List<Product> list = this.getProductsFromMysql();
                // 3.采用redis list数据结构的lpush命令来实现存储
                redisTemplate.delete(JHS_KEY);
                // 4.加入最新的数据给redis
                redisTemplate.opsForList().leftPushAll(JHS_KEY, list);
                // 5.暂停1分钟,间隔1分钟执行一次,模拟聚划算一天执行的参加活动的品牌
                try {
                    TimeUnit.MINUTES.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"t1").start();

    }

    /**
     * 差异失效时间
     */
    @PostConstruct         // 测试双缓存
    public void initJHSAB(){
        log.info("启动AB的定时器 天猫聚划算模拟开始 ===============");
        // 1.用线程模拟定时任务,后台任务定时将mysql里面的特价商品刷新的redis
        new Thread(() -> {
            while (true){
                // 2.模拟从mysql查到数据,加到redis并返回给页面
                List<Product> list = this.getProductsFromMysql();

                // 3.先更新B缓存且让B缓存过期时间超过缓存A时间,如果A突然失效了还有B兜底,防止击穿
                redisTemplate.delete(JHS_KEY_B);
                redisTemplate.opsForList().leftPushAll(JHS_KEY_B, list);
                redisTemplate.expire(JHS_KEY_B, 86410L, TimeUnit.SECONDS);
                // 4.再更新A缓存
                redisTemplate.delete(JHS_KEY_A);
                redisTemplate.opsForList().leftPushAll(JHS_KEY_A, list);
                redisTemplate.expire(JHS_KEY_A, 86400L, TimeUnit.SECONDS);

                // 5.暂停1分钟,间隔1分钟执行一次,模拟聚划算一天执行的参加活动的品牌
                try {
                    TimeUnit.MINUTES.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"t1").start();
    }
}

JHSProductController类

import com.xfcy.entities.Product;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.util.CollectionUtils;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;


@RestController
@Slf4j
@Api(tags = "聚划算商品列表接口")
public class JHSProductController {

    public static final String JHS_KEY = "jhs";
    public static final String JHS_KEY_A = "jhs:a";
    public static final String JHS_KEY_B = "jhs:b";

    @Autowired
    private RedisTemplate redisTemplate;

    /**
     * 分页查询:在高并发情况下,只能走redis查询,走db必定会把db打垮
     * @param page
     * @param size
     * @return
     */
    @RequestMapping(value = "/product/find", method = RequestMethod.GET)
    @ApiOperation("聚划算案例,每次1页每页5条显示")
    public List<Product> find(int page, int size) {
        List<Product> list = null;

        long start = (page - 1) * size;
        long end = start + size - 1;

        try {
            // 采用redis list结构里面的range命令来实现加载和分页
            list = redisTemplate.opsForList().range(JHS_KEY, start, end);
            if (CollectionUtils.isEmpty(list)) {
                // TODO 走mysql查询

            }
            log.info("参加活动的商家: {}", list);
        }catch (Exception e){
            // 出异常了,一般redis宕机了或者redis网络抖动导致timeout
            log.error("jhs  exception{}", e);
            e.printStackTrace();
            // 。。。重试机制 再次查询mysql

        }
        return list;
    }


    @RequestMapping(value = "/product/findAB", method = RequestMethod.GET)
    @ApiOperation("AB双缓存架构,防止热点key突然消失")
    public List<Product> findAB(int page, int size) {
        List<Product> list = null;

        long start = (page - 1) * size;
        long end = start + size - 1;

        try {
            // 采用redis list结构里面的range命令来实现加载和分页
            list = redisTemplate.opsForList().range(JHS_KEY_A, start, end);
            if (CollectionUtils.isEmpty(list)) {
                log.info("-----A缓存已经过期或活动结束了,记得人工修补,B缓存继续顶着");

                // A没有来找B
                list = redisTemplate.opsForList().range(JHS_KEY_B, start, end);

                if (CollectionUtils.isEmpty(list)){
                    // TODO 走mysql查询
                }
            }
            log.info("参加活动的商家: {}", list);
        }catch (Exception e){
            // 出异常了,一般redis宕机了或者redis网络抖动导致timeout
            log.error("jhs  exception{}", e);
            e.printStackTrace();
            // 。。。重试机制 再次查询mysql
        }
        return list;
    }
}

在这里插入图片描述

8.5 总结

缓存问题产生原因解决方案
缓存更新不一致数据变更、缓存时效性同步更新、失效更新、异步更新、定时更新
缓存不一致同步更新失败、异步更新增加重试、补偿任务、最终一致
缓存穿透恶意攻击空对象缓存、bloomFilter 过滤器
缓存击穿热点key失效互斥更新、随即退避、差异失效时间
缓存雪崩缓存挂掉快速失败熔断、主从模式、集群模式

在这里插入图片描述

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

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

相关文章

【Linux网络编程三】Udp套接字编程(简易版服务器)

【Linux网络编程三】Udp套接字编程(简易版服务器&#xff09; 一.创建套接字二.绑定网络信息1.构建通信类型2.填充网络信息①网络字节序的port②string类型的ip地址 3.最终绑定 三.读收消息1.服务器端接收消息recvfrom2.服务器端发送消息sendto3.客户端端发送消息sendto4.客户端…

前端小案例——滚动文本区域(HTML+CSS, 附源码)

一、前言 实现功能: 这个案例实现了一个具有滚动功能的文本区域&#xff0c;用于显示长文本内容&#xff0c;并且可以通过滚动条来查看完整的文本内容。 实现逻辑&#xff1a; 内容布局&#xff1a;在<body>中&#xff0c;使用<div>容器创建了一个类名为listen_t…

【Java程序设计】【C00242】基于Springboot的冬奥会科普平台(有论文)

基于Springboot的冬奥会科普平台&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 项目简介 这是一个基于Springboot的冬奥会科普平台 本系统分为系统功能模块以及管理员功能模块。 系统功能模块&#xff1a;登录进入冬奥会科普平台可以查看首页&…

ToF 测距传感器 VL6180 测量范围修改(软件 I2C)

TOF 测距传感器 VL6180 传感器修改测量范围 ...... by 矜辰所致前言 之前写过一篇关于ToF 测距传感器 VL6180 使用的文章&#xff1a; ToF 测距传感器 VL6180 使用踩坑记&#xff08;软件 I2C&#xff09; 之后有粉丝问我如何修改测量距离&#xff0c;当时我只回答让粉丝去…

【c语言】简单贪吃蛇的实现

目录 一、游戏说明 ​编辑 二、地图坐标​ ​编辑 三、头文件 四、蛇身和食物​ 五、数据结构设计​ 蛇节点结构如下&#xff1a; 封装一个Snake的结构来维护整条贪吃蛇&#xff1a;​ 蛇的方向&#xff0c;可以一一列举&#xff0c;使用枚举&#xff1a; 游戏状态&a…

QT播放gstreamer命令(三)---使用QMediaPlayer

前文: 因为之前听说过,QMediaPlayer已经集成了gstreamer,但是并没有什么接口来例子来说明,根本看不出来有任何gstreamer的形式,于是在QT5助手里面搜了一下,发现确实有gstreamer的痕迹,但是例子写的极其拉胯,经过自己尝试,终于发现了一种直接使用QMediaPlayer播放gstre…

uniapp中组件库Mask 遮罩层 的使用方法

目录 #平台差异说明 #基本使用 #嵌入内容 #遮罩样式 #API #Props #Events #Slot 创建一个遮罩层&#xff0c;用于强调特定的页面元素&#xff0c;并阻止用户对遮罩下层的内容进行操作&#xff0c;一般用于弹窗场景 #平台差异说明 AppH5微信小程序支付宝小程序百度小程…

将链表反转

反转链表在解决需要从尾节点开始遍历到头节点的地方很实用&#xff0c;是一种常用的解题技巧。在反转时&#xff0c;我们可以考虑从前向后反转和从后向前反转两种方式。 法一&#xff1a;递归 每次将链表的头节点的下一个节点作为新的头节点&#xff0c;然后对剩余部分调用递归…

阿里云OSS对象存储

一、前言 阿里云对象存储OSS作用&#xff1a;用于存储图片、视屏、文件等数据。 参考阿里云文档地址&#xff1a;阿里云对象存储教程 二、总体思路 说明&#xff1a;客户端给服务端发送请求&#xff0c;获取policy和signature等数据&#xff08;服务端提供&#xff09;&#…

前端小案例——导航回顶部(HTML+CSS+JS, 附源码)

一、前言 实现功能&#xff1a; 这个案例实现了页面滚动到一定位置时显示"回到顶部"按钮&#xff0c;并且点击按钮能够平滑滚动回页面顶部的功能。 实现逻辑&#xff1a; 页面结构&#xff1a;通过HTML标签定义了页面的基本结构。页面主要由多个div.content组成&am…

MySQL-----初识

一 SQL的基本概述 基本概述 ▶SQL全称: Structured Query Language&#xff0c;是结构化查询语言&#xff0c;用于访问和处理数据库的标准的计算机语言。SQL语言1974年由Boyce和Chamberlin提出&#xff0c;并首先在IBM公司研制的关系数据库系统SystemR上实现。 ▶美国国家标…

如何解决 docker registry x509 证书不信任问题?

最近想尝试一下极狐GitLab&#xff08;可以理解为 GitLab 在中国的发行版&#xff09;内置的容器镜像仓库&#xff0c;这样就不用自己安装 Harbor 之类的了。于是找了个服务器安装了一个极狐GitLab 的私有化部署版本&#xff0c;安装过程可以参考过往的技术文章使用Omnibus 安装…

刨析数据结构(二)

&#x1f308;个人主页&#xff1a;小田爱学编程 &#x1f525; 系列专栏&#xff1a;数据结构————"带你无脑刨析" &#x1f3c6;&#x1f3c6;关注博主&#xff0c;随时获取更多关于数据结构的优质内容&#xff01;&#x1f3c6;&#x1f3c6; &#x1f600;欢迎…

Day06-Linux下目录命令讲解及重要文件讲解

Day06-Linux下目录命令讲解及重要文件讲解 1. Linux目录文件1.1 Linux系统目录结构介绍1.1.1 Linux与Windows目录结构对比 1.2 重要的Linux配置文件介绍1.2.1 /etc系统初始化及设置相关重要文件1.2.2 /usr目录的重要知识介绍------应用程序目录1.2.3 /var目录下的路径知识-----…

Qt|制作简单的不规则窗体

通常我们用到的对话框基本上都是规则的&#xff0c;在有些特殊情况下&#xff0c;也会使用到不规则窗口&#xff0c;那么该如何实现不规则窗体呢&#xff1f; 在MFC框架下很难实现&#xff0c;应该说是难的都想放弃&#xff0c;但是&#xff0c;Qt框架下提供了一个叫做setMask…

【JAVA】守护线程是什么?

&#x1f34e;个人博客&#xff1a;个人主页 &#x1f3c6;个人专栏&#xff1a;JAVA ⛳️ 功不唐捐&#xff0c;玉汝于成 目录 正文 我的其他博客 正文 在计算机编程中&#xff0c;守护线程&#xff08;daemon thread&#xff09;是一种在程序运行时在后台提供服务的线程…

【正点原子STM32连载】 第四十七章 FATFS实验 摘自【正点原子】APM32E103最小系统板使用指南

1&#xff09;实验平台&#xff1a;正点原子APM32E103最小系统板 2&#xff09;平台购买地址&#xff1a;https://detail.tmall.com/item.htm?id609294757420 3&#xff09;全套实验源码手册视频下载地址&#xff1a; http://www.openedv.com/docs/boards/xiaoxitongban 第四…

【模型微调】| 各类微调模型总结 P-Tuning,Prefix,P-tuning v2,LoRA

文章目录 1 微调背景1.1 Full fine-tuning 全参数微调&#xff08;FFT&#xff09;1.2 parameter-Efficient-fine-tuning 部分参数微调&#xff08;PEFT&#xff09; 2 提示词调整训练法2.1 P-Tuning2.2 Prefix2.3 P-Tuning v2 3 结构调整训练法3.1 Adapter tuning3.2 LoRA 微调…

【2024年美国大学生数学建模竞赛】F题非法的野生动物贸易 完整数据

小云更新了全网最全的F题数据 另外也为大家分享&#xff1a; 技术文档&#xff0c;包括问题分析、建立模型、求解结果等&#xff0c;配套有思路分析视频、代码讲解视频。美赛官方限制总页数为25页&#xff0c;我们的思路长度为35页以上。所有模型都有求解代码和指标&#xff0…

数字化转型:企业适应新常态的关键之举_光点科技

在全球商业环境不断演变和技术日新月异的背景下&#xff0c;数字化转型已经成为企业不可回避的课题。它不仅关乎企业的未来生存与发展&#xff0c;更是适应新常态、提升竞争力的关键之举。但是&#xff0c;数字化转型并非一夜之间可以完成的任务&#xff0c;它需要全面的策略规…