这篇文章主要介绍了Spring注解和同步锁不能同步问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
结论:如果在service层的方法上同时使用事务和同步锁无法保证数据同步。
- @Service
- public class ServiceImpl{
- private static Lock lock = new ReentrantLock(false);
- @Transactional(rollbackFor = Exception.class)
- public void update() {
- try {
- lock.lock();
- ... ...
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- lock.unlock();
- }
- }
- }
上面这个例子无法保证数据的一致性,synchronized 同理。
原因:
根据spring的AOP的特性,会在update方法之前开启事务,之后再加锁,当锁住的代码执行完成后,再提交事务。
由于lock代码块执行是在事务之内执行的,在代码块执行完时,事务还未提交,因此其它线程进入synchronized代码块后,读取的数据库数据不是最新的(脏读)。
解决方案:
1.在还没有开启事务之前就加同步锁,用加锁的方法调用加事务的方法
- @Service
- public class ServiceImpl{
- private static Lock lock = new ReentrantLock(false);
- public void update1() {
- try {
- lock.lock();
- update2();
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- lock.unlock();
- }
- }
- @Transactional(rollbackFor = Exception.class)
- public void uodate2() {
- ... ...
- }
- }
2.把锁放到上一层
- @Controller
- public class TestController{
- @Autowired
- private IServiceImpl serviceImpl;
- private static Lock lock = new ReentrantLock(false);
- public String test() {
- try {
- lock.lock();
- serviceImpl.update();
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- lock.unlock();
- }
- }
- }
- @Service
- public class ServiceImpl{
- @Transactional(rollbackFor = Exception.class)
- public void update() {
- ... ...
- }
- }