专业游戏门户,分享手游网游单机游戏百科知识攻略!

嗨游网
嗨游网

SpringBoot AOP Redis如何实现延时双删功能

来源:小嗨整编  作者:小嗨  发布时间:2024-03-16 08:34
摘要:一、业务场景在多线程并发情况下,假设有两个数据库修改请求,为保证数据库与redis的数据一致性,修改请求的实现中需要修改数据库后,级联修改redis中的数据。请求一:a修改数据库数据b修改redis数据请求二:c修改数据库数据d修改re...
一、业务场景

在多线程并发情况下,假设有两个数据库修改请求,为保证数据库与redis的数据一致性,修改请求的实现中需要修改数据库后,级联修改redis中的数据。请求一:a修改数据库数据 b修改redis数据请求二:c修改数据库数据 d修改redis数据并发情况下就会存在a —> c —> d —> b的情况(一定要理解线程并发执行多组原子操作执行顺序是可能存在交叉现象的)

SpringBoot AOP Redis如何实现延时双删功能

1、此时存在的问题

A修改数据库的数据最终保存到了Redis中,C在A之后也修改了数据库数据。

此时出现了Redis中数据和数据库数据不一致的情况,在后面的查询过程中就会长时间去先查Redis,从而出现查询到的数据并不是数据库中的真实数据的严重问题。

2、解决方案

在使用Redis时,需要保持Redis和数据库数据的一致性,最流行的解决方案之一就是延时双删策略。注意:要知道经常修改的数据表不适合使用Redis,因为双删策略执行的结果是把Redis中保存的那条数据删除了,以后的查询就都会去查询数据库。所以Redis使用的是读远远大于改的数据缓存。延时双删方案执行步骤

1> 删除缓存2> 更新数据库3> 延时500毫秒 (根据具体业务设置延时执行的时间)4> 删除缓存

3、为何要延时500毫秒?

我们需要在第二次Redis删除之前完成数据库的更新操作。假象一下,如果没有第三步操作时,有很大概率,在两次删除Redis操作执行完毕之后,数据库的数据还没有更新,此时若有请求访问数据,便会出现我们一开始提到的那个问题。

4、为何要两次删除缓存?

如果我们没有第二次删除操作,此时有请求访问数据,有可能是访问的之前未做修改的Redis数据,删除操作执行后,Redis为空,有请求进来时,便会去访问数据库,此时数据库中的数据已是更新后的数据,保证了数据的一致性。

二、代码实践

1、引入Redis和SpringBoot AOP依赖

      org.springframework.boot      spring-boot-starter-data-redis      org.springframework.boot      spring-boot-starter-aop
登录后复制

2、编写自定义aop注解和切面

ClearAndReloadCache延时双删注解

/** *延时双删 **/@Retention(RetentionPolicy.RUNTIME)@Documented@Target(ElementType.METHOD)public @interface ClearAndReloadCache {    String name() default "";}
登录后复制

ClearAndReloadCacheAspect延时双删切面

@Aspect@Componentpublic class ClearAndReloadCacheAspect {@Autowiredprivate StringRedisTemplate stringRedisTemplate;/*** 切入点*切入点,基于注解实现的切入点  加上该注解的都是Aop切面的切入点**/@Pointcut("@annotation(com.pdh.cache.ClearAndReloadCache)")public void pointCut(){}/*** 环绕通知* 环绕通知非常强大,可以决定目标方法是否执行,什么时候执行,执行时是否需要替换方法参数,执行完毕是否需要替换返回值。* 环绕通知第一个参数必须是org.aspectj.lang.ProceedingJoinPoint类型* @param proceedingJoinPoint*/@Around("pointCut()")public Object aroundAdvice(ProceedingJoinPoint proceedingJoinPoint){    System.out.println("----------- 环绕通知 -----------");    System.out.println("环绕通知的目标方法名:" + proceedingJoinPoint.getSignature().getName());    Signature signature1 = proceedingJoinPoint.getSignature();    MethodSignature methodSignature = (MethodSignature)signature1;    Method targetMethod = methodSignature.getMethod();//方法对象    ClearAndReloadCache annotation = targetMethod.getAnnotation(ClearAndReloadCache.class);//反射得到自定义注解的方法对象    String name = annotation.name();//获取自定义注解的方法对象的参数即name    Set keys = stringRedisTemplate.keys("*" + name + "*");//模糊定义key    stringRedisTemplate.delete(keys);//模糊删除redis的key值    //执行加入双删注解的改动数据库的业务 即controller中的方法业务    Object proceed = null;    try {        proceed = proceedingJoinPoint.proceed();    } catch (Throwable throwable) {        throwable.printStackTrace();    }    //开一个线程 延迟1秒(此处是1秒举例,可以改成自己的业务)    // 在线程中延迟删除  同时将业务代码的结果返回 这样不影响业务代码的执行    new Thread(() -> {        try {            Thread.sleep(1000);            Set keys1 = stringRedisTemplate.keys("*" + name + "*");//模糊删除            stringRedisTemplate.delete(keys1);            System.out.println("-----------1秒钟后,在线程中延迟删除完毕 -----------");        } catch (InterruptedException e) {            e.printStackTrace();        }    }).start();    return proceed;//返回业务代码的值    }}
登录后复制

3、application.yml

server:  port: 8082spring:  # redis setting  redis:    host: localhost    port: 6379  # cache setting  cache:    redis:      time-to-live: 60000 # 60s  datasource:    driver-class-name: com.mysql.cj.jdbc.Driver    url: jdbc:mysql://localhost:3306/test    username: root    password: 1234# mp settingmybatis-plus:  mapper-locations: classpath*:com/pdh/mapper/*.xml  global-config:    db-config:      table-prefix:  configuration:    # log of sql    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl    # hump    map-underscore-to-camel-case: true
登录后复制

4、user_db.sql脚本

用于生产测试数据

DROP TABLE IF EXISTS `user_db`;CREATE TABLE `user_db`  (  `id` int(4) NOT NULL AUTO_INCREMENT,  `username` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL,  PRIMARY KEY (`id`) USING BTREE) ENGINE = InnoDB AUTO_INCREMENT = 8 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;-- ------------------------------ Records of user_db-- ----------------------------INSERT INTO `user_db` VALUES (1, '张三');INSERT INTO `user_db` VALUES (2, '李四');INSERT INTO `user_db` VALUES (3, '王二');INSERT INTO `user_db` VALUES (4, '麻子');INSERT INTO `user_db` VALUES (5, '王三');INSERT INTO `user_db` VALUES (6, '李三');
登录后复制

5、UserController

/** * 用户控制层 */@RequestMapping("/user")@RestControllerpublic class UserController {    @Autowired    private UserService userService;    @GetMapping("/get/{id}")    @Cache(name = "get method")    //@Cacheable(cacheNames = {"get"})    public Result get(@PathVariable("id") Integer id){        return userService.get(id);    }    @PostMapping("/updateData")    @ClearAndReloadCache(name = "get method")    public Result updateData(@RequestBody User user){        return userService.update(user);    }    @PostMapping("/insert")    public Result insert(@RequestBody User user){        return userService.insert(user);    }    @DeleteMapping("/delete/{id}")    public Result delete(@PathVariable("id") Integer id){        return userService.delete(id);    }}
登录后复制

6、UserService

/** * service层 */@Servicepublic class UserService {    @Resource    private UserMapper userMapper;    public Result get(Integer id){        LambdaQueryWrapper wrapper = new LambdaQueryWrapper<>();        wrapper.eq(User::getId,id);        User user = userMapper.selectOne(wrapper);        return Result.success(user);    }    public Result insert(User user){        int line = userMapper.insert(user);        if(line > 0)            return Result.success(line);        return Result.fail(888,"操作数据库失败");    }    public Result delete(Integer id) {        LambdaQueryWrapper wrapper = new LambdaQueryWrapper<>();        wrapper.eq(User::getId, id);        int line = userMapper.delete(wrapper);        if (line > 0)            return Result.success(line);        return Result.fail(888, "操作数据库失败");    }    public Result update(User user){        int i = userMapper.updateById(user);        if(i > 0)            return Result.success(i);        return Result.fail(888,"操作数据库失败");    }}
登录后复制三、测试验证

1、ID=10,新增一条数据

2、第一次查询数据库,Redis会保存查询结果

3、第一次访问ID为10

4、第一次访问数据库ID为10,将结果存入Redis

5、更新ID为10对应的用户名(验证数据库和缓存不一致方案)

数据库和缓存不一致验证方案:

打个断点,模拟A线程执行第一次删除后,在A更新数据库完成之前,另外一个线程B访问ID=10,读取的还是旧数据。

利用第二次删除,在根据业务场景设定合适的延迟时间后,待两次删除缓存成功后,Redis的输出结果将为空。读取的都是数据库真实数据,不会出现读缓存和数据库不一致情况。

四、代码工程

核心代码红色方框所示

以上就是SpringBoot AOP Redis如何实现延时双删功能的详细内容,更多请关注易企推科技其它相关文章!


本文地址:网络百科频道 https://www.eeeoo.cn/wangluo/1148965.html,嗨游网一个专业手游免费下载攻略知识分享平台,本站部分内容来自网络分享,不对内容负责,如有涉及到您的权益,请联系我们删除,谢谢!


网络百科
小编:小嗨整编
相关文章相关阅读
  • 电脑如何设置密码锁屏(电脑如何设置密码的方法)?

    电脑如何设置密码锁屏(电脑如何设置密码的方法)?

    电脑如何设置密码锁屏(电脑如何设置密码的方法)?随着电脑在日常生活中的普及,个人信息的安全性越来越受到重视。设置密码锁屏是保护电脑隐私的有效方法。本文将为您详细介绍如何在电脑上设置密码锁屏,帮助您轻松实现电脑安全防护。一、Windows系统...

  • 360皮肤中心如何关闭(360皮肤中心怎样删除)?

    360皮肤中心如何关闭(360皮肤中心怎样删除)?

    360皮肤中心如何关闭(360皮肤中心怎样删除)?360皮肤中心是360公司推出的一款电脑软件,用户可以通过它来更改电脑桌面图标、界面风格等,从而实现个性化设置。然而,有些用户可能觉得360皮肤中心的使用体验不佳,想要关闭或删除它。360皮...

  • 密码本忘记密码怎么开锁(密码本忘记密码如何打开)?

    密码本忘记密码怎么开锁(密码本忘记密码如何打开)?

    密码本忘记密码怎么开锁(密码本忘记密码如何打开)?随着信息安全意识的提高,越来越多的人开始使用密码本来记录重要的账号和密码。然而,忘记密码本密码的情况也时有发生。本文将为您介绍几种忘记密码本密码时的开锁方法。密码本忘记密码怎么开锁1.按照提...

  • cad迷你看图软件如何打印图纸(cad迷你看图软件下载官网)?

    cad迷你看图软件如何打印图纸(cad迷你看图软件下载官网)?

    cad迷你看图软件如何打印图纸(cad迷你看图软件下载官网)?AutoCAD(AutoComputerAidedDesign)是美国Autodesk公司首次于1982年生产的自动计算机辅助设计软件,用于二维绘图、详细绘制、设计文档和基本三维...

  • apk软件在哪下载(apk软件如何安装)?

    apk软件在哪下载(apk软件如何安装)?

    apk软件在哪下载(apk软件如何安装)?APK是AndroidPackage的缩写,它是一种以Zip格式打包的安装文件。APK文件包含了应用程序的代码、资源文件和清单文件等。当您点击安装APK文件时,安卓系统会自动解压并安装该应用程序到您...

  • 魔兽世界如何重置副本进度(魔兽世界如何重置副本命令)

    魔兽世界如何重置副本进度(魔兽世界如何重置副本命令)

    魔兽世界如何重置副本进度(魔兽世界如何重置副本命令)下面就让我们一起来探讨一下魔兽世界中副本重置的方法吧。魔兽世界如何重置副本进度为了确保游戏过程的流畅与稳定,小编强烈建议玩家们在重置副本前先使用奇游加速工具优化网络环境。这样不仅可以减少卡...

  • maya软件主要功能是什么(maya软件是免费的吗)?

    maya软件主要功能是什么(maya软件是免费的吗)?

    maya软件主要功能是什么(maya软件是免费的吗)?Maya是一款三维动画软件,它的功能强大,工作灵活性高,制作效率高,渲染真实感极强,是电影级别的高端制作软件,而在学习了Maya应用之后,可以从事影视模型、广告、角色动画、电影特效等多种...

  • b站直播软件有哪些功能(b站直播软件怎么设置)?

    b站直播软件有哪些功能(b站直播软件怎么设置)?

    b站直播软件有哪些功能(b站直播软件怎么设置)?B站在电商板块布局已久。2017年8月,B站上线“会员购”。随后几年,B站陆续增加UP主店铺、第三方商品外链等功能,逐步搭建电商生态。B站UP主后台上线全新的直播带货中控平台,用于实时监测带货...

  • 周排行
  • 月排行
  • 年排行

精彩推荐