redis分布式锁踩坑|spring 事务踩坑

问题记录


问题记录:redisson分布式锁和@Transactional在并发场景中的Bug

详细描述:线程A抢占到线程B未提交事务但已释放的锁的,以及线程A读取到未提交的数据,导致线程A处理用户流水明细记录时,出现流水账目错误。

业务场景:房间内用户可以相互打赏礼物,在高频刷礼物的场景下,导致用户余额扣增错误

解决方法:

        1、通过生产环境的日志进行原因分析

        2、猜测可能出现的错误原因

  • 事务隔离机制导致的释放锁后,事务未提交,导致读取到旧数据
  • 其它线程把正在执行任务线程的锁给释放了

        3、模拟生产环境的业务代码,进行场景还原

        4、确定问题是因为抢到锁之后,读取到了未提交的事务数据,问题进行修改

问题解决方案

原业务逻辑伪代码:        

@Transactional(rollbackFor = Exception.class, propagation = Propagation.REQUIRES_NEW)
public Action consume() {
    
    //xxxx逻辑处理
    Business giftShareAmountMessage = new Business();
    serviceA.
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot 中使用 Redis 实现分布式锁,你可以按照以下步骤进行: 1. 添加 Redis 依赖 在 `pom.xml` 文件中添加 Redis 相关的依赖: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> ``` 2. 配置 Redis 连接 在 `application.properties` 或 `application.yml` 文件中配置 Redis 连接信息: ```properties spring.redis.host=127.0.0.1 spring.redis.port=6379 ``` 3. 创建分布式锁工具类 ```java import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.stereotype.Component; import java.util.concurrent.TimeUnit; @Component public class RedisDistributedLock { @Autowired private RedisTemplate<String, String> redisTemplate; public boolean lock(String lockKey, String requestId, long expireTime) { Boolean success = redisTemplate.opsForValue().setIfAbsent(lockKey, requestId, expireTime, TimeUnit.MILLISECONDS); return success != null && success; } public boolean releaseLock(String lockKey, String requestId) { String value = redisTemplate.opsForValue().get(lockKey); if (value != null && value.equals(requestId)) { return redisTemplate.delete(lockKey); } return false; } } ``` 4. 在需要加锁的地方使用分布式锁 ```java import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; @RestController @RequestMapping("/example") public class ExampleController { @Autowired private RedisDistributedLock distributedLock; @GetMapping("/lock") public String lockExample() { String lockKey = "exampleLock"; String requestId = UUID.randomUUID().toString(); long expireTime = 5000; // 锁的过期时间,单位为毫秒 // 尝试获取锁 boolean lockSuccess = distributedLock.lock(lockKey, requestId, expireTime); if (lockSuccess) { try { // 执行业务逻辑 Thread.sleep(2000); return "Success"; } catch (InterruptedException e) { e.printStackTrace(); } finally { // 释放锁 distributedLock.releaseLock(lockKey, requestId); } } return "Failed"; } } ``` 在上述代码中,首先创建了一个 `RedisDistributedLock` 的工具类,用来进行锁的获取和释放操作。然后,在需要加锁的地方调用 `lock()` 方法尝试获取锁,如果获取成功,则执行业务逻辑;最后,在业务逻辑执行完成后,调用 `releaseLock()` 方法释放锁。 注意:在上述示例中,使用了 `RedisTemplate` 作为 Redis 的操作模板,你可以根据实际情况进行调整和优化。另外,还可以对分布式锁进行进一步的优化,例如使用 Lua 脚本实现原子性操作等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值