springboot整合redis+lua实现getdel操作保证原子性

  1. 原始代码

脚本逻辑先获取redis的值,判断是否等于期望值。 条件成立则删除,不成立则返回0

Lua 复制代码
if redis.call('get', KEYS[1]) == ARGV[1] then
    return redis.call('del', KEYS[1])
end
return 0
  1. 测试代码

根据上面的逻辑加了测试, 在判断成功后等待5秒后执行删除操作。同时开启另外一个线程去修改这个key的值, 发现修改的线程一直阻塞。直到等待的线程5秒结束后且完成删除操作后,修改的线程才会执行。

原因:

redis是单线程执行,

Redis会将整个Lua脚本作为一个原子操作执行,这意味着脚本中的多个命令要么全部执行成功,要么全部失败。这有助于维护数据的原子性和一致性。

Redis执行Lua脚本时,它会自动锁定相关的键,以防止其他客户端对这些键进行并发修改。这确保了在脚本执行期间对键的访问是互斥的。

注意: 如果脚本中涉及多个key的操作并redis部署的是cluster模式时, 需要额外确认多个key都在一个slot中。还好我这就1个key!!!略过.............

java 复制代码
String str_script =
            // 1. 判断key的值是否等于期望值,等于则进行删除,不相等则直接结束
            "if (redis.call('get', KEYS[1]) == ARGV[1]) then\n" +
            "    local start = redis.call('TIME')[1] * 1000 + redis.call('TIME')[2] / 1000\n" +
            "    while (redis.call('TIME')[1] * 1000 + redis.call('TIME')[2] / 1000 - start) < 5000 do\n" +
            "        print('等待5秒结束.....')"+
            "    end\n"+
            // 2. 值判断成功,等待5秒执行删除操作
            "    return redis.call('del', KEYS[1])\n" +
            "end\n" +
            "return 0";
    public String getAndDelete(String key, String expectedValue) {
        log.info("start..................");
        DefaultRedisScript script = new DefaultRedisScript(str_script, Long.class);
        Object res = redisTemplate.execute(script, Collections.singletonList(key), expectedValue);
        log.info("end..................");
        return res.toString();
    }
相关推荐
熙胤5 分钟前
springboot与springcloud对应版本
java·spring boot·spring cloud
于眠牧北10 分钟前
ubuntu22.04在docker中安装redis6.2.x并配置远程连接
运维·redis·docker·容器
J2虾虾10 分钟前
SpringBoot 中给 @Autowired 搭配 @Lazy
java·spring boot·后端
智_永无止境1 小时前
Spring Boot 动态多数据源:核心思路与关键考量
spring boot
czlczl200209251 小时前
Redis分布式缓存与持久化 杂知识
redis·分布式·缓存
sheji34161 小时前
【开题答辩全过程】以 基于springboot的健身预约系统的设计与实现为例,包含答辩的问题和答案
java·spring boot·后端
没有bug.的程序员2 小时前
500个微服务上云全线假死:Spring Boot 3.2 自动配置底层的生死狙击
java·spring boot·微服务·kubernetes·自动配置
夜郎king2 小时前
实战教程:Leaflet+SpringBoot 实现地图任意点位点击查看时间功能
spring boot·webgis 时区生成·java时区可视化
陈皮糖..2 小时前
Docker Compose 学习之多容器应用编排与运维实践 —— 基于 Nginx+MySQL+Redis 服务栈的部署与管理
运维·redis·学习·mysql·nginx·docker
老毛肚2 小时前
Redis八股
数据库·redis·缓存