使用DB::transaction()可确保数据库操作原子性,如余额扣减与记录写入同时成功或回滚;支持手动事务控制,适用于复杂逻辑;需避免耗时操作、合理使用行锁,并注意事务生命周期与队列重试问题。

在 Laravel 中使用数据库事务可以确保一组数据库操作要么全部成功,要么全部失败回滚,从而保证数据的一致性。特别是在处理多表写入、更新或删除等关键业务逻辑时,事务非常有用。
1. 使用 DB::transaction() 开启事务
最常见的方式是使用 DB::transaction() 方法包裹需要原子执行的代码块。如果代码块内抛出异常,Laravel 会自动触发回滚;如果没有异常,则自动提交事务。
use Illuminate\Support\Facades\DB;DB::transaction(function () {
$user = DB::table('users')->where('id', 1)->lockForUpdate()->first();
if ($user->balance throw new \Exception('余额不足');
}
DB::table('users')->where('id', 1)->decrement('balance', 50);
DB::table('transactions')->insert([
'user_id' => 1,
'amount' => -50,
'created_at' => now(),
'updated_at' => now()
]);
});
上面的例子中,如果余额不足抛出异常,两个数据库操作都会被回滚。
2. 手动控制事务(begin、commit、rollback)
你也可以手动开启事务,适用于更复杂的控制逻辑。
try {
DB::table('users')->where('id', 1)->decrement('balance', 50);
DB::table('logs')->insert([
'action' => 'deduct_balance',
'user_id' => 1,
'created_at' => now(),
'updated_at' => now()
]);
DB::commit(); // 提交事务
} catch (\Exception $e) {
DB::rollBack(); // 回滚事务
// 可以记录日志或重新抛出异常
report($e);
}
这种方式适合需要捕获异常并进行额外处理的场景。
3. 注意事项与最佳实践
- 事务中尽量避免耗时操作(如远程请求),以免长时间锁表影响性能
- 使用 lockForUpdate() 可防止并发修改导致的数据不一致
- 事务不能跨 HTTP 请求,只在当前请求生命周期内有效
- 在队列任务中使用事务时,注意任务失败重试可能引发重复提交问题
基本上就这些。合理使用 Laravel 的事务机制,能有效保障关键业务的数据完整性。不复杂但容易忽略细节。










