如何使用ThinkPHP进行MySQL事务处理及操作技巧详解?
- 内容介绍
- 文章标签
- 相关推荐
本文共计1096个文字,预计阅读时间需要5分钟。
ThinkPHP与MySQL事务可用,但默认不生效——关键卡在三点:
Db::transaction() 闭包方式为什么最安全?
它自动封装了 startTrans()、commit() 和 rollback(),异常一抛就回滚,不用手写 try-catch。
- 闭包内任何地方 throw
\Exception或发生未捕获错误,事务自动回滚 - 闭包执行完无异常,自动提交,代码干净不易漏
- 不支持跨连接事务(比如同时操作两个不同数据库),否则会报错或静默失败
- 示例:
Db::transaction(function () { Db::table('order')->insert(['uid' => 100, 'status' => 'created']); Db::table('stock')->where('id', 1)->dec('num', 1); // 这里如果 stock 不足,dec 返回 false,但不会自动中断 // 必须手动判断并 throw,否则仍会 commit if (Db::table('stock')->where('id', 1)->value('num') < 0) { throw new \Exception('库存不足'); } });
手动事务 Db::startTrans() 容易在哪几步失效?
手动控制看似灵活,但每一步都可能断链:开启、判断、提交、回滚,缺一不可,且顺序不能乱。
本文共计1096个文字,预计阅读时间需要5分钟。
ThinkPHP与MySQL事务可用,但默认不生效——关键卡在三点:
Db::transaction() 闭包方式为什么最安全?
它自动封装了 startTrans()、commit() 和 rollback(),异常一抛就回滚,不用手写 try-catch。
- 闭包内任何地方 throw
\Exception或发生未捕获错误,事务自动回滚 - 闭包执行完无异常,自动提交,代码干净不易漏
- 不支持跨连接事务(比如同时操作两个不同数据库),否则会报错或静默失败
- 示例:
Db::transaction(function () { Db::table('order')->insert(['uid' => 100, 'status' => 'created']); Db::table('stock')->where('id', 1)->dec('num', 1); // 这里如果 stock 不足,dec 返回 false,但不会自动中断 // 必须手动判断并 throw,否则仍会 commit if (Db::table('stock')->where('id', 1)->value('num') < 0) { throw new \Exception('库存不足'); } });
手动事务 Db::startTrans() 容易在哪几步失效?
手动控制看似灵活,但每一步都可能断链:开启、判断、提交、回滚,缺一不可,且顺序不能乱。

