【已解决】MySQL 事务回滚机制失效之误用 truncate 删除表数据

🎉在工作场景中遇到这样一个需求,对表中的数据进行更新,更新的步骤是:首先,删除表中所有的数据记录;之后,再将更新过后的数据插入到表中。由于系统业务逻辑相关的原因,在数据插入时可能会出现异常,因此当出现异常而终止插入过程,需要能够将事务进行回滚,恢复之前删除的数据。

✨这里肯定是需要使用事务进行控制了,那么在使用 Spring 框架提供的声明式事务注解方式@Transactional ,并确认避免了事务失效的几种情况之后,在测试接口时事务还是没有生效。

✨以下是主要的相关方法实现:

@Transactional(rollbackFor = Exception.class)
public void runDeptTask() {
	//to do something
	try {
		deptMapper.deleteAll();
		//to do something
	} catch (Exception e) {
		e.printstack();
		// due to the exception was swallowed, rollbacking manually 
		TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
	}
}

✨在确认 Java 代码层面并没有问题之后,便开始从数据库层面开始排查问题。首先,检查是否使用了支持事务的存储引擎,确认 MySQL 数据库使用了 InnoDB 引擎并开启了事务机制。之后,在检查 SQL 语句时发现,删除数据时的语法使用了 truncate,而不是使用 delete 语法。

✨这时便回忆起,为了保证事务的原子性,InnoDB 引擎在实际进行记录的增删改操作时,都需要先把 undo 日志记下来,便于之后进行回滚。这里的增删改对应的语法是:insertdeleteupdate没有对使用 truncate 删除的 undo 日志。这便是发生问题的所在之处了,因此需要将删除时使用的语法改为 delete,这样才能使得事务回滚起效,问题得到解决。

这里顺便提一下为什么当时使用 truncate,主要是因为使用 truncate 不会写入 undo 日志,既降低了服务器硬盘资源的消耗,又避免了因写入日志操作带来的时间消耗,因此 truncate 删除表数据的效率非常高,特别是当删除大量数据时,truncatedelete 所需要的时间差异尤为明显。

✨更改过后的 SQL 语句如下:

DeptMapper.xml

<delete id = "deleteAll">
	delete from sys_dept
</delete>
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ReadThroughLife

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值