[案例] jpa和Transactional使用造成锁超时

1 篇文章 0 订阅

 一、问题背景

在用jpa中的saveAndFlush方法做保存时,由于保存时间过长(超过50秒),而其他线程对相同数据做保存而产生的锁超时和事务重启问题,报错如下

o.h.engine.jdbc.spi.SqlExceptionHelper   : 
Lock wait timeout exceeded; try restarting transaction
o.h.i.ExceptionMapperStandardImpl        : 
HHH000346: Error during managed flush [org.hibernate.exception.LockAcquisitionException: could not execute statement]

二、Jpa保存方法save和saveAndFlush的区别

1.在saveAndFlush()上,此命令中的更改将立即刷新到DB。

2.使用save(),就不一定了,它可能只暂时保留在内存中,直到发出flush或commit命令。但是要注意的是,即使在事务中刷新了更改但是未提交它们,这些更改对于外部事务仍然不可见,直到,提交这个事务

三、Jpa的保存方法在事务中所造成的锁超时

在我们执行一次定时任务,定时任务需要时间是100秒,其中对医生1修改保存时方法是saveAndFlush,使用saveAndFlush的话,执行的sql立即发送到数据,进行数据锁定;当用户刚好对医生1进行修改,这是数据已经被锁定,需要等到定时任务执行完或者锁超时时中断程序才能有结果返回

 @GetMapping("/testSave")
    @Transactional()
    public void testSave() throws InterruptedException {
        Doctor one = doctorRepository.getOne(1L);
        one.setTitle("222");
        Doctor save = doctorRepository.save(one);
//        Thread.sleep(10000);
        System.out.println(save.toString());
        System.out.println(save.getTitle());
    }

    @GetMapping("/testSaveAndFlush")
    @Transactional()
    public void testSaveAndFlush() throws InterruptedException {
        Optional<Doctor> byId = doctorRepository.findById(1L);
        if (byId.isPresent()) {
            Doctor one = byId.get();
            one.setTitle("111");
            Doctor save = doctorRepository.saveAndFlush(one);
            Thread.sleep(100000);
            System.out.println(save.toString());
            System.out.println(save.getTitle());
        }

    }

四、解决方法

1.事务细化,仅在提交数据时加上事务

2.数据分批提交,将大的数据集分成几个小的数据集处理,然后手动提交事务

2.数据分批提交,将大的数据集分成几个小的数据集处理,然后手动提交事务

2.数据分批提交,将大的数据集分成几个小的数据集处理,然后手动提交事务

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值