当前位置:   article > 正文

@Transactional 中使用线程锁导致了锁失效,震惊我一整年!_transactional使用线程锁导致了

transactional使用线程锁导致了

一、引出问题

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

在此举例说明。

1、数据准备

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

图片

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

图片

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

1、不加锁

  1. // service代码
  2. public void test() {
  3.     // 简单的select + update 模拟业务场景
  4.     Model model = mapper.choseOne("99");
  5.  // 实现 level -- 操作
  6.     Model updater = new Model();
  7.     updater.setId("99");
  8.     updater.setLevel(model.getLevel() - 1);
  9.     mapper.updateOne(updater);
  10. }

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

图片

2、使用锁

  1. // service代码
  2. private Lock lock = new ReentrantLock();
  3. public void test() {
  4.  try {
  5.      //加锁
  6.      lock.lock();
  7.      // 简单的select + update 模拟业务场景
  8.      Model model = mapper.choseOne("99");
  9.  
  10.   // 实现 level -- 操作
  11.      Model updater = new Model();
  12.      updater.setId("99");
  13.      updater.setLevel(model.getLevel() - 1);
  14.      mapper.updateOne(updater);
  15.  } finally {
  16.        lock.unlock(); // 解锁
  17.     }
  18. }

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

图片

3、使用锁+@Transactional

  1. // service代码
  2. private Lock lock = new ReentrantLock();
  3. @Transactional
  4. public void test() {
  5.  try {
  6.      //加锁
  7.      lock.lock();
  8.      // 简单的select + update 模拟业务场景
  9.      Model model = mapper.choseOne("99");
  10.  
  11.   // 实现 level -- 操作
  12.      Model updater = new Model();
  13.      updater.setId("99");
  14.      updater.setLevel(model.getLevel() - 1);
  15.      mapper.updateOne(updater);
  16.  } finally {
  17.        lock.unlock(); // 解锁
  18.     }
  19. }

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

图片

4、问题分析

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

通过下面的图理解一下:

图片

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

三、解决方案

1、@Transactional单独一个方法

  1. private Lock lock = new ReentrantLock();
  2. @Transactional
  3. public void test1() {
  4.     // 简单的select + update 模拟业务场景
  5.     Model model = mapper.choseOne("99");
  6.  // 实现 level -- 操作
  7.     Model updater = new Model();
  8.     updater.setId("99");
  9.     updater.setLevel(model.getLevel() - 1);
  10.     mapper.updateOne(updater);
  11. }
  12. @Autowired
  13. @Lazy
  14. private CommonService commonService;
  15. public void test() {
  16.     try {
  17.         // 加锁
  18.         lock.lock();
  19.         // 自己注入自己,以使用到其代理类
  20.         commonService.test1();
  21.     } finally {
  22.         lock.unlock(); // 解锁
  23.     }
  24. }

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

图片

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

2、使用编程式事务

  1. // service代码
  2. private Lock lock = new ReentrantLock();
  3. @Autowired
  4. private PlatformTransactionManager transactionManager;
  5. public void test() {
  6.  try {
  7.      //加锁
  8.      lock.lock();
  9.      // 编程式事务
  10.         TransactionStatus status = transactionManager.getTransaction(new DefaultTransactionDefinition());
  11.         
  12.      // 简单的select + update 模拟业务场景
  13.      Model model = mapper.choseOne("99");
  14.  
  15.   // 实现 level -- 操作
  16.      Model updater = new Model();
  17.      updater.setId("99");
  18.      updater.setLevel(model.getLevel() - 1);
  19.      mapper.updateOne(updater);
  20.      
  21.   // 在锁中提交
  22.         transactionManager.commit(status);
  23.  } finally {
  24.        lock.unlock(); // 解锁
  25.     }
  26. }

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

图片

最后说一句(求关注!别白嫖!)

如果这篇文章对您有所帮助,或者有所启发的话,求一键三连:点赞、转发、在看。

关注公众号:woniuxgg,在公众号中回复:笔记  就可以获得蜗牛为你精心准备的java实战语雀笔记,回复面试、开发手册、有超赞的粉丝福利!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/酷酷是懒虫/article/detail/1020913
推荐阅读
相关标签
  

闽ICP备14008679号