生产问题一:redis锁处理幂等性失效

伪代码:

java 复制代码
    @Transactional(rollbackFor = Exception.class)
    public void add(User user) {
        String key = "key";
        RLock lock = redissonClient.getLock(key);
        lock.lock();
        try {
            long count = userMapper.selectCount(user);
            if (count == 0) {
                userMapper.insert(user);
            }
        } catch (Exception e) {
            log.error("add user error", e);
        } finally {
            lock.unlock();
        }
        System.out.println("插入成功");
    }

问题分析:

如果有两个线程a,b。如果a线程释放锁后,退出方法前,让出时间片,由于方法未执行完,此时事务没有提交,那么b线程在去数据库查询的时候仍然出来count为0,执行了insert操作,两个线程执行完该方法提交事务,此时数据库中会增加两条数据,幂等性失效。

注:a事务未提交,查出来count仍然为0的原因是,mysql默认的事物隔离级别是可重复读,因此无法读取为提交的数据

解决方式:

第一种:

java 复制代码
    //@Transactional(rollbackFor = Exception.class)
    public void add(User user) {
        String key = "key";
        RLock lock = redissonClient.getLock(key);
        lock.lock();
        try {
            long count = userMapper.selectCount(user);
            if (count == 0) {
                userMapper.insert(user);
            }
        } catch (Exception e) {
            log.error("add user error", e);
        } finally {
            lock.unlock();
        }
        System.out.println("插入成功");
    }

去掉@Transactional注解,这样在a线程在insert时候就会自动提交事务,a释放锁后,b在查询时候,count不等0,不执行插入操作。

方法二:

java 复制代码
public class Test {

    @Resource
    private UserMapper userMapper;

    @Resource
    private UserService userService;

    @Resource
    private RedissonClient redissonClient;

    public void test(User user) {
        String key = "key";
        RLock lock = redissonClient.getLock(key);
        lock.lock();
        try {
            userService.addOk(user);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }

    @Transactional(rollbackFor = Exception.class)
    public void addOk(User user) {
        long count = userMapper.selectCount(user);
        if (count == 0) {
            userMapper.insert(user);
        }
    }
}

a提交事务后再释放锁后,b在查询时候,count不等0,不执行插入操作。
绝大数情况下,为了防止重复数据的产生,我们都会在表中加唯一索引,这是一个非常简单,并
且有效的方案。 虽说抛异常对数据来说没有影响,不会造成错误数据。但是为了保证接口幂等性,我们需要对该 异常进行捕获,然后返回成功。
扩展:
事务在生产实践中经常犯的错误:
1.事务范围:应该加入事务的代码未加入到事务中
2.事务大小:事务过大,是否有必要拆解小事务(如何优化),拆解后一致性问题。
3.传播范围: a.多线程之间不可传播;b.多个方法内如果异常被捕获,事务会被被标记为异常事务,不可以再次提交(虽然不影响数据,但是有报错信息)

相关推荐
黄雪超13 分钟前
JVM——打开JVM后门的钥匙:反射机制
java·开发语言·jvm
有梦想的攻城狮21 分钟前
spring中的@RabbitListener注解详解
java·后端·spring·rabbitlistener
李斯维23 分钟前
循序渐进 Android Binder(二):传递自定义对象和 AIDL 回调
android·java·android studio
androidwork25 分钟前
OkHttp 3.0源码解析:从设计理念到核心实现
android·java·okhttp·kotlin
程序员岳焱27 分钟前
Java 程序员成长记(二):菜鸟入职之 MyBatis XML「陷阱」
java·后端·程序员
我命由我1234528 分钟前
Spring Boot 项目集成 Redis 问题:RedisTemplate 多余空格问题
java·开发语言·spring boot·redis·后端·java-ee·intellij-idea
面朝大海,春不暖,花不开28 分钟前
Spring Boot消息系统开发指南
java·spring boot·后端
程序员岳焱30 分钟前
Java 程序员成长记(三):菜鸟入职之@Transactional「罢工」
java·后端·编程语言
Rocky40138 分钟前
JAVAEE->多线程:锁策略
java·开发语言·jvm
白宇横流学长38 分钟前
基于J2EE架构的在线考试系统设计与实现【源码+文档】
java·架构·java-ee