加入收藏 | 设为首页 | 会员中心 | 我要投稿 常州站长网 (https://www.0519zz.com/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 站长学院 > MySql教程 > 正文

「MySQL数据库」事务嵌套的解决之道

发布时间:2018-08-19 12:16:11 所属栏目:MySql教程 来源:编码之道
导读:技术沙龙 | 邀您于8月25日与国美/AWS/转转三位专家共同探讨小程序电商实战 前言 我们在开发一个复杂的系统时可能经常出现这样的场景:比如,A函数中调用了B函数,而A函数和B函数同时都使用了事务,这样就出现了事务嵌套。在MySQL的官方文档中有明确的说明My
技术沙龙 | 邀您于8月25日与国美/AWS/转转三位专家共同探讨小程序电商实战

「MySQL数据库」事务嵌套的解决之道

 前言

我们在开发一个复杂的系统时可能经常出现这样的场景:比如,A函数中调用了B函数,而A函数和B函数同时都使用了事务,这样就出现了事务嵌套。在MySQL的官方文档中有明确的说明MySQL是不支持嵌套事务的:

Transactions cannot be nested. This is a consequence of the implicit commit performed for any current transaction when you issue a START TRANSACTION statement or one of its synonyms.

那我们该如何解决MySQL的事务嵌套问题呢?

解决方法

目前,在PHP圈有两种比较通用的解决方法,一种是以Doctrine为代表的,设置回滚点的解决方法,一种是以Laravel为代表的,控制事务次数的解决方法。

Doctrine的解决方法

Doctrine解决方法的核心就是对回滚点的控制,如下:

「MySQL数据库」事务嵌套的解决之道

Doctrine中开启事务的方法

「MySQL数据库」事务嵌套的解决之道

Doctrine中事务回滚的方法

「MySQL数据库」事务嵌套的解决之道

Doctrine中事务提交的方法

Doctrine用一个_transactionNestingLevel来标识当前嵌套的级别,如果是1,也就是还没有嵌套,那就用默认的方法执行一下START TRANSACTION就ok了;如果大于1,也就是有嵌套的时候,它会帮我们创建一个savepoint。这个savepoint可以理解为一个事务记录点,当需要回滚时我们可以只回滚到这个点。

Laravel的解决方法

相对Doctrine而言,Laravel的解决方法稍微简单粗暴,它巧妙的使用了一个 transactions属性来记录了调用事务的次数。在事务开启,事务提交和事务回滚时,先判断transactions的属性值,只有当transactions的属性值为1时,才进行事务操作。如下:

「MySQL数据库」事务嵌套的解决之道

在开启事务时,我们先判断当前有几个事务,如果是第一个,ok,事务开始,否则就啥都不做。

「MySQL数据库」事务嵌套的解决之道

在事务提交时,也判断当前事务个数,如果是第一个,ok,提交事务,否则,就只将transactions属性值减一

「MySQL数据库」事务嵌套的解决之道

在事务回滚时,同样先判断当前事务个数,如果是第一个,ok,回滚事务,同时将transactions属性值置为0,否则,就只将transactions属性值减一。

在Laravel的解决方法中,在嵌套的内层里面实际上是木有真正的事务的,只有最外层一个整体的事务,虽然简单粗暴,但是也解决了在内层新建一个事务时会造成commit的问题。

【编辑推荐】

  1. 时间序列数据库为何会突然成为宠儿?
  2. 30个MySQL千万级大数据SQL查询优化技巧详解
  3. 对数据库MySQL中的JOIN详解
  4. MySQL/InnoDB乐观锁悲观锁共享锁、排它锁行锁表锁死锁概念的理解
  5. MySQL在删除表时I/O错误原因分析
【责任编辑:庞桂玉 TEL:(010)68476606】
点赞 0

(编辑:常州站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    热点阅读