项目实战典型案例1——redis只管存不管删除 让失效时间删除的问题

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: 项目实战典型案例1——redis只管存不管删除 让失效时间删除的问题

redis只管存不管删除 让失效时间删除的问题

一:背景介绍

本篇博客是对项目开发中出现的redis只管存不管删除 让失效时间删除的问题进行的总结和改进。目的是将经历转变为自己的经验。通过博客的方式分享给大家,大家一起共同进步和提高。

代码逻辑

1.通过redis查询该课程下所有班级的信息

2.如果没有从redis中查询到数据就从数据库中查询并存入到redis中一份
存在的问题

3. 并没有在更新课程下班级数据的时候删除缓存,会导致如果我更新了该课程下的班级数据,会导致缓存中的数据与数据中的数据不一致,当我再次查询数据的时候,如果我的key键没有过期会直接从redis中查询数据,但是查询的数据是没有更新之前的数据。如果我key键正好过期了才会从数据库中查询最新的数据,并更新到缓存中。

二:思路&方案

保证缓存数据与数据库中数据同步的方案。

  1. 查询的时候如果缓存存在则从缓存中获取,如果缓存不存在则从数据库中获取数据,并将获取到的数据存储在缓存中。
  2. 更新数据的时候删除缓存。

三:代码模拟

通过java maven项目模拟redis只管存不管删除 让失效时间删除的问题 进行错误示范正确示范

代码环境:java maven项目、mysql,redis

pom.xml文件

 <dependencies>
        <dependency>
            <groupId>redis.clients</groupId>
            <artifactId>jedis</artifactId>
            <version>3.3.0</version>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.16</version>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.20</version>
            <scope>compile</scope>
        </dependency>
    </dependencies>

1.错误示范

通过班级id查询课程名称

package org.example.controller;
import org.example.dao.UserInfoDao;
import org.example.entity.CourseConfiguration;
import redis.clients.jedis.Jedis;
/**
 * @author : [WangWei]
 * @version : [v1.0]
 * @className : UserInfoController
 * @description : [描述说明该类的功能]
 * @createTime : [2022/11/30 11:43]
 * @updateUser : [WangWei]
 * @updateTime : [2022/11/30 11:43]
 * @updateRemark : [描述说明本次修改内容]
 */
public  class SelectCourseNameByClassId {
    /*
     * @description:通过班级id查询课程名称
     * @author: wangwei
     * @date: 2023/3/4 14:24
     * @param: [args]
     * @return: void
     **/
    public static void main(String[] args) {
        //构造jedis,连接redis 并设置超时时间为100秒,在jedis的默认构造方法中,超时的时间一般被默认设置为2000毫秒,也就是2秒。
        Jedis jedis=new Jedis("ip",端口号,100000);
        jedis.auth("密码");
        CourseConfiguration courseConfiguration=null;
        //查询redis
        String courseName = jedis.get("WangWei:ClassId");
        if(courseName==""||courseName==null){
            //从数据库中查询
            UserInfoDao userInfoDao=new UserInfoDao();
             courseConfiguration = userInfoDao.selectCourseNameByClassId(55801751);
            //存入redis 并设置过期时间60秒
            jedis.set("WangWei:ClassId",courseConfiguration.getCourse_name());
            jedis.expire("WangWei:ClassId",60);
            System.out.println(courseConfiguration.getCourse_name());
        }else {
            System.out.println(courseName);
        }
    }
}

执行结果



通过班级id修改课程名称(并没有删除对应缓存)

package org.example.controller;
import org.example.dao.UserInfoDao;
import org.example.entity.CourseConfiguration;
import org.example.utils.RedisConfig;
import redis.clients.jedis.Jedis;
/**
 * @author : [WangWei]
 * @version : [v1.0]
 * @className : UpdateCourseNameByClassId
 * @description : [描述说明该类的功能]
 * @createTime : [2023/3/4 14:26]
 * @updateUser : [WangWei]
 * @updateTime : [2023/3/4 14:26]
 * @updateRemark : [描述说明本次修改内容]
 */
public class UpdateCourseNameByClassId {
    /*
     * @description:通过通过班级id修改班级名称
     * @author: wangwei
     * @date: 2023/3/4 14:27
     * @param: [args]
     * @return: void
     **/
    public static void main(String[] args) {
        //修改班级名称
        UserInfoDao userInfoDao=new UserInfoDao();
        userInfoDao.updateCourseNameByClassId(55801751,"ARPro计算机思维课程-开发班课");
//        //连接redis
//        Jedis jedis=new Jedis("82.157.199.3",6379,100000);
//        jedis.auth("000415");
//        //删除key键
//        jedis.del("WangWei:ClassId");
        System.out.println("修改成功!");
    }
}

执行结果


这个时候我们再执行 通过班级id查询课程名称的接口由于WangWei:ClassId这个key键还没有过期,会发现查询的还是之前没有修改的课程名称。数据出现不一致。



当key键WangWei:ClassId过期时由于查询的是数据库中的数据,所以数据是一致的。



2.正确示范

  1. 查询的时候如果缓存存在则从缓存中获取,如果缓存不存在则从数据库中获取数据,并将获取到的数据存储在缓存中。
  2. 更新数据的时候删除缓存。

在错误示范的更新接口上添加删除缓存的代码

package org.example.controller;
import org.example.dao.UserInfoDao;
import redis.clients.jedis.Jedis;
/**
 * @author : [WangWei]
 * @version : [v1.0]
 * @className : UpdateCourseNameByClassId
 * @description : [描述说明该类的功能]
 * @createTime : [2023/3/4 14:26]
 * @updateUser : [WangWei]
 * @updateTime : [2023/3/4 14:26]
 * @updateRemark : [描述说明本次修改内容]
 */
public class UpdateCourseNameByClassId {
    /*
     * @description:通过通过班级id修改班级名称
     * @author: wangwei
     * @date: 2023/3/4 14:27
     * @param: [args]
     * @return: void
     **/
    public static void main(String[] args) {
        //修改班级名称
        UserInfoDao userInfoDao=new UserInfoDao();
        userInfoDao.updateCourseNameByClassId(55801751,"ARPro计算机思维课程-开发班课");
        //连接redis
        Jedis jedis=new Jedis("82.157.199.3",6379,100000);
        jedis.auth("000415");
        //删除key键
        jedis.del("WangWei:ClassId");
        System.out.println("修改成功!");
    }
}

通过以上的缓存同步测试,就可以保证缓存中的数据与实际数据库中的数据保持一致。

四:总结

1.出现这次redis只管存不管删除的问题,在于对于redis在实际项目中的使用不够熟悉了解,不清楚缓存同步的策略。推荐需要对redis参照官网进行系统性的学习,以及与其他高人进行讨论交流。
2.其实对于此次问题,如果没有遇到这个案例,自己也很有可能会同样的获取,由于自己本身对于redis的使用没有涉及到,以及考虑到redis缓存同步。

五:升华

通过这次的案例分析,对于认知上需要提升的点,自己也有了针对性提升对于知识认知上的方式方法。如思维导图,三篇读数法,参照官网对例如redis进行快速浏览,通过思维导图宏观了解,提升认知,不怕不知道就怕不知道。

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
6月前
|
存储 缓存 NoSQL
Redis缓存设计典型问题
缓存穿透 缓存穿透是指查询一个根本不存在的数据, 缓存层和存储层都不会命中, 通常出于容错的考虑, 如果从存储层查不到数据则不写入缓存层。缓存穿透将导致不存在的数据每次请求都要到存储层去查询, 失去了缓存保护后端存储的意义。
|
2月前
|
NoSQL Linux Redis
linux安装单机版redis详细步骤,及python连接redis案例
这篇文章提供了在Linux系统中安装单机版Redis的详细步骤,并展示了如何配置Redis为systemctl启动,以及使用Python连接Redis进行数据操作的案例。
55 2
|
22天前
|
消息中间件 NoSQL Kafka
大数据-116 - Flink DataStream Sink 原理、概念、常见Sink类型 配置与使用 附带案例1:消费Kafka写到Redis
大数据-116 - Flink DataStream Sink 原理、概念、常见Sink类型 配置与使用 附带案例1:消费Kafka写到Redis
86 0
|
5月前
|
JSON NoSQL Redis
|
6月前
|
缓存 NoSQL Java
Redis7的10大应用场景和案例解析
你在项目中使用 Redis 实现了什么应用场景,欢迎一起跟 V 哥讨论。同时也做个小调查,朋多少兄弟是需要了解 Redis 核心源码的,人多的话,下一篇 V 哥写 Redis7的源码分析,人少的话就算了,感谢。
132 0
|
6月前
|
NoSQL Redis
Redis企业项目实战--登录校验拦截器
Redis企业项目实战--登录校验拦截器
|
6月前
|
存储 监控 NoSQL
【Redis技术专区】「优化案例」谈谈使用Redis慢查询日志以及Redis慢查询分析指南
【Redis技术专区】「优化案例」谈谈使用Redis慢查询日志以及Redis慢查询分析指南
159 0
|
6月前
|
缓存 NoSQL 前端开发
【Redis技术专区】「实战案例」谈谈使用Redis缓存时高效的批量删除的几种方案
【Redis技术专区】「实战案例」谈谈使用Redis缓存时高效的批量删除的几种方案
141 0
|
6月前
|
NoSQL Java 数据库
优惠券秒杀案例 - CAS、Redis+Lua脚本解决高并发并行
优惠券秒杀案例 - CAS、Redis+Lua脚本解决高并发并行
297 0
|
6月前
|
消息中间件 存储 NoSQL
【Redis项目实战】使用Springcloud整合Redis分布式锁+RabbitMQ技术实现高并发预约管理处理系统
【Redis项目实战】使用Springcloud整合Redis分布式锁+RabbitMQ技术实现高并发预约管理处理系统