1-为啥会失效
@Transactional的实现是基于Spring的AOP。Spring会对其业务方法进行动态代理,执行前开始事务,然后再执行业务代码,最后提交事务。大致流程如下图所示
假设线程1线抢到锁,在解锁这个时候,这个时候事务还没有提交,数据库数据还没有更新。然后线程2获取到了锁,读取的数据可能是旧数据,出现了锁失效的情况。
synchronized和分布式锁都同理会失效!,最本质的原因:解锁在提交事务之前了。
2-解决方案
- 在Controller层加锁,注意锁粒度…避免长事务「长事务可能会增加死锁的概率…然后数据库连接池数量被打满… 」
- 在service层里再单独抽个方法,上锁操作、释放锁操作 夹着事务方法