redis分布式事务

1. 使用RedisTemplate.opsForValue().setIfAbsent(key, value, time, TimeUnit.SECONDS)

这种方法是基于Redis的SET命令的NX(Not eXists)选项和EX(expiry time)选项来实现的。它尝试设置一个键值对,只有当键不存在时才进行设置,并且可以指定该键的过期时间。

  • 优点

    • 使用简单,直接通过Spring Data Redis的RedisTemplate提供的API即可实现。
    • 原子操作:由于是基于Redis命令实现,这个操作是原子的,避免了并发问题。
  • 局限性

    • 仅适用于单一键的操作。如果需要对多个键进行原子性操作,这种方法就无能为力了。
    • 逻辑复杂度有限,不适合处理更复杂的事务逻辑。

2. 使用redis.call() Lua脚本

Lua是一种轻量级的脚本语言,Redis支持在服务器端执行Lua脚本。通过将一系列Redis命令组织成Lua脚本并一次性发送到Redis服务器执行,可以确保这些操作的原子性。

  • 优点

    • 原子性: 整个Lua脚本作为一个整体被执行,中间不会被其他客户端的请求打断,从而保证了操作的原子性。
    • 灵活性: 可以编写复杂的逻辑来处理多个键的事务,甚至实现条件判断、循环等控制结构。
    • 减少网络开销: 所有操作通过一次网络往返完成,相比多次独立的Redis命令调用更高效。
  • 使用示例:

    Lua 复制代码
    local key = KEYS[1]
    local value = ARGV[1]
    local time = tonumber(ARGV[2])
    
    if redis.call('setnx', key, value) == 1 then
        redis.call('expire', key, time)
        return true
    else
        return false
    end

    然后通过RedisTemplate.execute()方法执行此脚本。

  • 局限性

    • 需要编写Lua脚本,对于不熟悉Lua的开发者来说有一定的学习成本。
    • 脚本的调试相对困难,特别是对于复杂的逻辑。

区别总结

  • 操作复杂度setIfAbsent适合简单的单一键操作,而Lua脚本则能处理更复杂的多键事务和逻辑。
  • 原子性和性能:两者都能提供原子性操作,但Lua脚本在处理多步骤事务时能减少网络往返,提高性能。
  • 易用性setIfAbsent方法使用简单,无需编写额外脚本;Lua脚本虽然灵活但增加了开发和维护的复杂度。

根据实际需求选择合适的方法:如果只是简单的设置键值对并确保其唯一性,setIfAbsent足够使用;而对于涉及多键的复杂事务逻辑,则推荐使用Lua脚本。

相关推荐
金仓拾光集1 小时前
筑牢风控生命线:金仓数据库替代MongoDB,重构证券融资融券业务的数据基石
数据库·mongodb·信创·1024程序员节·kingbasees·国产化替代
那我掉的头发算什么1 小时前
【数据库】navicat的下载以及数据库约束
android·数据库·数据仓库·sql·mysql·数据库开发·数据库架构
纪伊路上盛名在1 小时前
如何批量获取蛋白质序列的所有结构域(domain)数据-2
数据库·人工智能·机器学习·统计·计算生物学·蛋白质
2301_772093563 小时前
高并发webserver_interview
运维·服务器·数据库·后端·网络协议·mysql·wireshark
yumgpkpm3 小时前
CMP(类ClouderaCDP7.3(404次编译) )完全支持华为鲲鹏Aarch64(ARM),粉丝数超过200就开源下载
hive·hadoop·redis·mongodb·elasticsearch·hbase·big data
大G的笔记本4 小时前
MySQL 大表查询优化、超大分页处理、SQL 慢查询优化、主键选择
数据库·sql·mysql
Lear5 小时前
Redis 持久化机制
数据库
儒道易行5 小时前
【攻防实战】Redis未授权RCE联动metasploit打穿三层内网(上)
数据库·redis·网络安全·缓存
小宁爱Python5 小时前
从零搭建 RAG 智能问答系统 6:Text2SQL 与工作流实现数据库查询
数据库·人工智能·python·django
yjsstar5 小时前
数据库MySQL基础
数据库·mysql