本篇内容主要讲解“怎么解决Spring注解和同步锁不能同步的问题”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么解决Spring注解和同步锁不能同步的问题”吧!
创新互联是一家集网站建设,西安企业网站建设,西安品牌网站建设,网站定制,西安网站建设报价,网络营销,网络优化,西安网站推广为一体的创新建站企业,帮助传统企业提升企业形象加强企业竞争力。可充分满足这一群体相比中小企业更为丰富、高端、多元的互联网需求。同时我们时刻保持专业、时尚、前沿,时刻以成就客户成长自我,坚持不断学习、思考、沉淀、净化自己,让我们为更多的企业打造出实用型网站。
结论:如果在service层的方法上同时使用事务和同步锁无法保证数据同步。
@Servicepublic 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.在还没有开启事务之前就加同步锁,用加锁的方法调用加事务的方法
@Servicepublic 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.把锁放到上一层
@Controllerpublic 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(); } }}@Servicepublic class ServiceImpl{ @Transactional(rollbackFor = Exception.class) public void update() { ... ... }}
到此,相信大家对“怎么解决Spring注解和同步锁不能同步的问题”有了更深的了解,不妨来实际操作一番吧!这里是创新互联网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!