spring使用@Transactional开启事务后,2次查询结果相同

数据库中存在一张表,用于存储每个调查问卷的填报编号,填报编号连续且唯一

问卷填报后,在最终提交时, 调用service的接口,获取填报编号

    @Override
    @Transactional
	public synchronized String getQuestionAnswerNumber(String code) {
        Integer no = genNumberMapper.getQuestionAnswerNoByCode(code);
        if(no == null){
    		genNumberMapper.insertQuestionAnswerNoByCode(code);
    		no = 0;
    	}
    	GenNumber genNumber = new GenNumber();
    	genNumber.setStub(code);
    	genNumber.setNo(no + 1);
    	genNumberMapper.setQuestionAnswerNoByCode(genNumber);
    	return String.format("%06d", (no + 1));
	}

代码编写流程主要是,通过code关键字,获取填号,然后将编号加1存入到数据库,这样下一个人获取编号时,能够按照顺序递增。为了避免多线程情况下,填报编号错乱,使用synchronized关键字,为方法加锁。

然而在生产环境中,并发量大的情况下,导出数据时,填报编号还是重复了。。。。

我们已经在方法生加了锁,为何编号还会出现重复?

我们在controller中开了多个线程,每个线程中循环100次,调用这个方法。结果还真出现了编号重复的情况。

@RequestMapping("/testSave")
    @ResponseBody
    @Transactional
    public void testSave() {
        new Thread(new Runnable(){
            @Override
            public void run() {
                for(int i=0; i<100; i++){
                    genNumberService.getQuestionAnswerNumber("a");
                }
            }}).start();

        new Thread(new Runnable(){
            @Override
            public void run() {
                for(int i=0; i<100; i++){
                    genNumberService.getQuestionAnswerNumber("a");
                }
            }}).start();
    }

经过分析,spring 在使用@Transactional关键字开启事务时,在方法执行完毕前,事务没有提交,数据会缓存在mybatis的一级缓存中,当方法执行完后,数据再进行提交。可我们为方法加了锁,为何还是会产生重复的数据呢,经过推测与测试,可能是获取填报编号的方法已经执行了,但事物并没有提交,这个时候另一个线程已经已经进来,导致获取的数据不准确。

验证方式

1.去掉@Transactional关键字,不开启事务。 -- 没有出现重复的

2.在controller层的方法上加锁,service的方法上去掉锁。  -- 没有出现重复的

经过以上方式,我们可以得出结论:spring在事务提交之前,另一个线程已经已经进来,导致获取的数据不准确。

项目采取的解决方案:在controller层的方法上加锁,service的方法上去掉锁。

补充:mybatis缓存机制

-----------------------------------------------------------------------------

感谢 “衣服架子”的评论

看到评论后,重新对问题进行了思考,应该是synchronized和@Transaction一起使用导致的。spring中对事务的处理,是采用动态代理的方式,事务里的方法是包含synchronized方法,导致有多个线程进入到事务中。

参考:https://www.cnblogs.com/Java3y/p/10392645.html

 

 

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
对于这个问题,我可以为您提供以下建议: 1. 在Spring Boot项目中使用@Transactional注解需要配置事务管理器。可以通过在配置文件中添加以下内容来配置使用事务管理器: spring.datasource.url=jdbc:mysql://localhost:3306/db_name spring.datasource.username=db_username spring.datasource.password=db_password #特别要注意这个属性: #spring.jpa.show-sql=true #transactionManager spring.jpa.hibernate.ddl-auto=create-drop spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.MySQL5InnoDBDialect spring.jpa.hibernate.naming.implicit-strategy=org.hibernate.boot.model.naming.ImplicitNamingStrategyLegacyJpaImpl spring.jpa.hibernate.naming.physical-strategy=org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl spring.jpa.hibernate.naming.implicit-strategy=org.hibernate.boot.model.naming.ImprovedNamingStrategy spring.jpa.properties.hibernate.format_sql=true spring.jpa.properties.hibernate.use_sql_comments=true 2. 如果您需要在不同的方法中开启和关闭事务,可以将@Transactional注解添加到相关方法上。如果希望整个类的方法都使用相同事务,则可以将@Transactional注解添加到类级别。 3. 还需要确保在pom.xml文件中添加了以下依赖项: <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-jpa</artifactId> </dependency> 以上仅供参考,具体实现会根据您的具体需求和项目情况而有所不同。希望可以帮助到您,谢谢!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值