场景是这样的,从数据库中查找,如果查到:将值加1再插入到数据库中;返回该值。如果没查到,往数据库中插入一条值为1的记录;返回1.
很自然的使用了synchronized来保证同步,同时方法上还加上了@Transactional注解处理事务。可是在做单元测试的时候,发现多线程环境下,出现了并发问题,比如开启10个线程同时调用该方法,部分线程返回的值是相同的,数据库中最终的值也不是10。
原因:spring@Transactional注解使用的是AOP来实现,也就是说被@Transactional注解的方法的事务是由spring生成的一个代理类来处理的,当一个线程执行完该方法并释放锁后,代理类还没有提交事务前,别的线程是有机会进入到该方法中的,这样一来,就有几率访问到过期的数据,从而导致并发问题。
解决办法:如果只是对一个表的更新,那么可以去掉@Transactional关键字,就不会有并发问题。