mybatis中commit和rollback使用不当造成事务泄露和锁超时问题总结

commit接口定义如下:
  /**
   * Flushes batch statements and commits database connection.
   * Note that database connection will not be committed if no updates/deletes/inserts were called.
   * To force the commit call {@link SqlSession#commit(boolean)}
   */
  void commit();
  /**
   * Flushes batch statements and commits database connection.
   * @param force forces connection commit
   */
  void commit(boolean force);
在使用这两个函数不当时,可能会造成事务泄露和锁超时问题。
1. 事务泄露场景
注意commit()注释提示部分,打开一个不自动提交事务的session,如果这个session在commit之前没有update/delete/insert执行语句时,session.commit()是不会提交事务的,此时要想提交事务需要使用commit(true)方法,否则在session.close()之前这个事务是一直没有释放的。

2. 锁超时问题
也许你会说针对没有更新语句的场景我们可以开启自动提交事务的session来避免,但是当需要select for update语句时候你是必须要开启非自动提交事务的session的。此时如果不使用session.commit(true),不仅会造成事务没有及时释放而且可能造成逻辑上的错误(锁超时问题)。
比如以下代码就会造成逻辑上的错误:
// 开启一个非自动提交事务session
SqlSession session1 = sessionFactory.openSession(false);
// select for update 语句锁住一条主键为1的记录
selectByIdForUpdate(session1, 1);
// 此处我们想提交事务,释放锁;但是事与愿违
session1.commit();
// 开启一个非自动提交事务session
SqlSession session2 = sessionFactory.openSession(false);
// 使用新的session锁住这条记录,此处会出现锁超时问题,
// 因为上个事务并没有真正的提交;这不是我们锁期望的
selectByIdForUpdate(session2, 1);
session2.commit();
// 关闭连接
session1.close();
session2.close();
rollback的使用和commit的类似,也应该注意这些问题。
综上所述:在使用commit和rollback的时候要注意这些问题,在没有确定session是否有更新语句的情况下,尽量使用commit(true)和rollback(true)函数。这些问题都是我们在工作过程中遇到了,仔细查看了源码和用户手册总结出的经验教训,希望对大家使用mybatis有一定帮助。
  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
SpringBoot整合MyBatis可以使用Spring的事务管理机制来实现事务的回滚和提交。 在SpringBoot,我们可以通过在方法上使用`@Transactional`注解来声明事务,用于标识需要进行事务管理的方法。当方法执行时,如果出现异常导致方法终止,则事务管理器会自动回滚事务,将所有数据库操作都撤销到事务开始前的状态。如果方法执行成功,则事务管理器会将所有数据库操作统一提交。 事务的回滚和提交是由Spring的事务管理器来实现的。Spring提供了几种常用的事务管理器实现,可以根据具体的项目需求选择合适的事务管理器。 其,如果要使用MyBatis事务管理器,需要在SpringBoot的配置文件(application.properties或application.yml)进行配置。具体配置项如下: ``` # 指定事务管理器类型为JDBC spring.datasource.type=org.apache.tomcat.jdbc.pool.DataSource # 开启事务支持 spring.datasource.tomcat.initSQL=getLockTimeout spring.datasource.url=jdbc:mysql://localhost:3306/mydb spring.datasource.username=root spring.datasource.password=root # 配置MyBatis的Mapper接口 mybatis.type-aliases-package=com.example.model mybatis.mapper-locations=classpath:mapper/*.xml # 配置事务管理器 mybatis.mapper-locations=classpath:mapper/*.xml ``` 配置完成后,就可以在需要进行事务管理的方法上添加`@Transactional`注解,即可实现事务的自动回滚和提交。 需要注意的是,事务的回滚和提交是根据方法的异常情况来决定的,只有在方法抛出运行时异常或Error时才会触发事务的回滚。如果在方法抛出的是受检异常(如IOException),则不会触发事务的回滚。如果需要手动触发事务的回滚,可以在方法捕获异常,并调用`TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()`方法来设置事务的回滚状态。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值