天天看点

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;  

继续阅读