• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
孙龙 程序员
少时总觉为人易,华年方知立业难
博客园    首页    新随笔    联系   管理    订阅  订阅
PHP中实现MySQL嵌套事务解决方案

在MySQL的官方文档中有明确的说明不支持嵌套事务

是在我们开发一个复杂的系统时难免会无意中在事务中嵌套了事务,比如A函数调用了B函数,A函数使用了事务,并且是在事务中调用了B函数,B函数也有一个事务,这样就出现了事务嵌套。这时候其实A的事务就意义不大了,为什么呢?上面的文档中就有提到,简单的翻译过来就是:

1. 当执行一个START TRANSACTION指令时,会隐式的执行一个commit操作。 所以我们就要在系统架构层面来支持事务的嵌套。

解决方法:

 要在开始事务的时候记录一个 count 值,每次开始事务 count + 1,同时调用 MySQL的开始事务。
提交事务时 count - 1,只有当 count = 0 时才调用 MySQL 的提交事务
回滚事务时 count 清 0,同时调用MySQL的回滚事务
aravel的解决方案laravel的处理方式相对简单粗暴一些,我们先来看下创建事务的操作:

public function beginTransaction()

{

++$this->transactions;
if ($this->transactions == 1)

{

$this->pdo->beginTransaction();

}

}

感觉如何?so easy吧?先判断当前有几个事务,如果是第一个,ok,事务开始,否则就啥都不做,那么为啥是啥都不做呢?继续往下看rollBack的操作:

public function rollBack()

{

if ($this->transactions == 1)

{

$this->transactions = 0;
$this->pdo->rollBack();

}

else

{

--$this->transactions;

}

}

明白了吧?只有当当前事务只有一个的时候才会真正的rollback,否则只是将计数做减一操作。这也就是为啥刚才说laravel的处理比较简单粗暴一些,在嵌套的内层里面实际上是木有真正的事务的,只有最外层一个整体的事务,虽然简单粗暴,但是也解决了在内层新建一个事务时会造成commit的问题。原理就是这个样子了,为了保持完整起见,把commit的代码也copy过来吧!
public function commit()  
  
{  
  
    if ($this->transactions == 1) $this->pdo->commit();  
    --$this->transactions;  
  
}  
 

本文来自博客园,作者:孙龙-程序员,转载请注明原文链接:https://www.cnblogs.com/sunlong88/articles/8672117.html

posted on 2018-03-29 20:11  孙龙-程序员  阅读(700)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3