这篇文章主要介绍了Spring注解和同步锁不能同步问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 结论:如果在ser
这篇文章主要介绍了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() { ... ... } }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持易盾网络。