@Transactional 中使用线程锁导致了锁失效,惊了!

a558b56185bf6f8ee14e9a49526b94e1.jpeg来源:juejin.cn/post/7311603432925102095

👉 欢迎加入小哈的星球 ,你将获得: 专属的项目实战 / Java 学习路线 / 一对一提问 / 学习打卡 /  每月赠书

新项目:仿小红书(微服务架构)正在更新中... , 全栈前后端分离博客项目 2.0 版本完结啦, 演示链接http://116.62.199.48/ 。全程手摸手,后端 + 前端全栈开发,从 0 到 1 讲解每个功能点开发步骤,1v1 答疑,直到项目上线。目前已更新了261小节,累计41w+字,讲解图:1806张,还在持续爆肝中.. 后续还会上新更多项目,目标是将Java领域典型的项目都整一波,如秒杀系统, 在线商城, IM即时通讯,Spring Cloud Alibaba 等等,戳我加入学习,已有1400+小伙伴加入(早鸟价超低)

9a68575ac687158892cfba3c5adc672f.gif

一、引出问题

很多小伙伴使用Spring事务时,为了省事都喜欢使用@Transactional。但是@Transactional配合锁,会导致一些预期之外的问题!

在此举例说明。

1、数据准备

我们将在该表中,实现level数据递减的并发操作。

cef12bf3655be08b17c4f8e05ead7256.png
图片

Controller中,简单模拟10个线程各自执行10次:

5a955c52816ab6730b1127846f077865.png
图片

二、@Transactional是如何导致锁失效的

1、不加锁

// service代码
public void test() {
    // 简单的select + update 模拟业务场景
    Model model = mapper.choseOne("99");

 // 实现 level -- 操作
    Model updater = new Model();
    updater.setId("99");
    updater.setLevel(model.getLevel() - 1);
    mapper.updateOne(updater);
}

执行结果:我们发现,level只扣减了26,说明存在并发问题!

93e4a3b48d59610f96528a6520bed239.png
图片

2、使用锁

// service代码
private Lock lock = new ReentrantLock();

public void test() {
 try {
     //加锁
     lock.lock();
     // 简单的select + update 模拟业务场景
     Model model = mapper.choseOne("99");
 
  // 实现 level -- 操作
     Model updater = new Model();
     updater.setId("99");
     updater.setLevel(model.getLevel() - 1);
     mapper.updateOne(updater);
 } finally {
       lock.unlock(); // 解锁
    }
}

执行结果:我们发现,使用锁是可以控制并发问题。

f729bcad6bdfeb3471f92f26c993a422.png
图片

3、使用锁+@Transactional

// service代码
private Lock lock = new ReentrantLock();

@Transactional
public void test() {
 try {
     //加锁
     lock.lock();
     // 简单的select + update 模拟业务场景
     Model model = mapper.choseOne("99");
 
  // 实现 level -- 操作
     Model updater = new Model();
     updater.setId("99");
     updater.setLevel(model.getLevel() - 1);
     mapper.updateOne(updater);
 } finally {
       lock.unlock(); // 解锁
    }
}

执行结果:我们发现,level只扣减了86!用了@Transactional之后,锁怎么就失效了呢!

6dbb3f2d5cc65f7274ab45351708beaa.png
图片

4、问题分析

我们都知道,@Transactional是通过使用AOP,在目标方法执行前后进行事务的开启和提交。所以,Lock锁住的代码,其实并没有包含住一整个事务!

通过下面的图理解一下:

c406a438bf3d33fb4ccefdbce71f6a5a.png
图片

当线程A将level设置为99时,此时锁已经释放了,但是事务还没提交!!线程B此时可以获取到锁并进行查询,查询出来的level还是线程A修改之前的100,所以出现了并发问题。

三、解决方案

1、@Transactional单独一个方法

private Lock lock = new ReentrantLock();
@Transactional
public void test1() {
    // 简单的select + update 模拟业务场景
    Model model = mapper.choseOne("99");

 // 实现 level -- 操作
    Model updater = new Model();
    updater.setId("99");
    updater.setLevel(model.getLevel() - 1);
    mapper.updateOne(updater);
}

@Autowired
@Lazy
private CommonService commonService;
public void test() {
    try {
        // 加锁
        lock.lock();
        // 自己注入自己,以使用到其代理类
        commonService.test1();
    } finally {
        lock.unlock(); // 解锁
    }
}

执行结果:没有并发问题出现!

ee1e218be9f72ec1fdfdd2ee80903f94.png
图片

或者直接在controller层加锁,也是一样的道理。

2、使用编程式事务

// service代码
private Lock lock = new ReentrantLock();
@Autowired
private PlatformTransactionManager transactionManager;
public void test() {
 try {
     //加锁
     lock.lock();
     // 编程式事务
        TransactionStatus status = transactionManager.getTransaction(new DefaultTransactionDefinition());
        
     // 简单的select + update 模拟业务场景
     Model model = mapper.choseOne("99");
 
  // 实现 level -- 操作
     Model updater = new Model();
     updater.setId("99");
     updater.setLevel(model.getLevel() - 1);
     mapper.updateOne(updater);
     
  // 在锁中提交
        transactionManager.commit(status);
 } finally {
       lock.unlock(); // 解锁
    }
}

执行结果:我们发现,将整个事务都锁住,就没问题了!

90bb3fdeca8ec689178ba437ec3be334.png
图片

👉 欢迎加入小哈的星球 ,你将获得: 专属的项目实战 / Java 学习路线 / 一对一提问 / 学习打卡 /  每月赠书

新项目:仿小红书(微服务架构)正在更新中... , 全栈前后端分离博客项目 2.0 版本完结啦, 演示链接http://116.62.199.48/ 。全程手摸手,后端 + 前端全栈开发,从 0 到 1 讲解每个功能点开发步骤,1v1 答疑,直到项目上线。目前已更新了261小节,累计41w+字,讲解图:1806张,还在持续爆肝中.. 后续还会上新更多项目,目标是将Java领域典型的项目都整一波,如秒杀系统, 在线商城, IM即时通讯,Spring Cloud Alibaba 等等,戳我加入学习,已有1400+小伙伴加入(早鸟价超低)

d50be06d87f8d7169c2d1c19143c71eb.gif

82e385e7ac288656164e0ff22bb9d231.jpeg

 
 

9cf75512e6881118b0fb101c1cbe2011.gif

 
 
 
 
1. 我的私密学习小圈子~
2. 面试官:Spring Boot 的启动原理是什么?
3. 如何优雅的实现在线人数统计功能?
4. 你还在使用 WebSocket 实现实时消息推送吗?
 
 
最近面试BAT,整理一份面试资料《Java面试BATJ通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。
获取方式:点“在看”,关注公众号并回复 Java 领取,更多内容陆续奉上。
PS:因公众号平台更改了推送规则,如果不想错过内容,记得读完点一下“在看”,加个“星标”,这样每次新文章推送才会第一时间出现在你的订阅列表里。
点“在看”支持小哈呀,谢谢啦
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值