laravel 并发控制写法-涉及资金


一、两种方案正面对比

方案 A:事务 + SELECT ... FOR UPDATE

复制代码
DB::transaction(function () {
    $user = DB::table('users')
        ->where('id', $id)
        ->lockForUpdate()
        ->first();

    if ($user->balance >= $amount) {
        DB::table('users')
            ->where('id', $id)
            ->update(['balance' => $user->balance - $amount]);
    }
});

方案 B:原子 UPDATE

复制代码
$affected = DB::update(
    'UPDATE users
     SET balance = balance - ?
     WHERE id = ? AND balance >= ?',
    [$amount, $id, $amount]
);

二、为什么方案 B 效率更高

1️⃣ SQL 次数:1 次 vs 2 次

方案 SQL
FOR UPDATE SELECT + UPDATE
原子 UPDATE UPDATE(1 次)

少一次:

  • 网络 IO

  • SQL 解析

  • 执行计划

  • 锁切换

👉 在高并发下差距非常明显


2️⃣ 锁的持有时间:极短

FOR UPDATE
  • 锁从 SELECT 开始

  • 一直到事务提交

  • 中间可能跑 PHP 逻辑、日志、异常处理

原子 UPDATE
  • 锁只在 一条 UPDATE 执行期间

  • 几乎是毫秒级

👉 锁时间越短,并发吞吐越高


3️⃣ 锁的范围更小、更可控

复制代码
WHERE id = ? AND balance >= ?
  • 主键精确命中

  • 只锁 1 行

  • 不产生多余间隙锁

而:

复制代码
SELECT ... FOR UPDATE
  • 如果条件不严谨

  • 可能锁多行 / 锁范围扩大

  • 更容易死锁


三、并发安全性:谁更强?

维度 FOR UPDATE 原子 UPDATE
防超扣
防并发
防逻辑漏洞 ⚠️ 依赖代码 ✅ 数据库保证
出错概率 较高 极低

👉 原子 UPDATE 把风险从「代码层」下沉到「数据库层」


四、那 FOR UPDATE 什么时候才值得用?

✅ 适合 FOR UPDATE 的场景

  1. 必须先读,再做复杂决策

    复制代码
    余额 + 状态 + 规则 + 多字段计算
  2. 要锁多行 / 多表

    • 订单 + 库存 + 账户

    • 且逻辑无法合并成一条 SQL

  3. 需要保证"读到的就是将要修改的"


❌ 不适合 FOR UPDATE 的场景

  • 单字段扣减(余额 / 库存 / 次数)

  • 能写成:

    复制代码
    UPDATE ... WHERE 条件
  • 高并发、低延迟要求

👉 这正是支付扣款场景


FINALLY:

💰 钱 / 库存 / 次数

原子 UPDATE

复制代码
UPDATE ... SET x = x - ? WHERE x >= ?

📦 复杂业务状态流转

transaction + FOR UPDATE

相关推荐
老邓计算机毕设1 天前
SSM智慧社区信息化服务平台4v5hv(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面
数据库·ssm 框架·智慧社区、·信息化平台
泉-java1 天前
第56条:为所有导出的API元素编写文档注释 《Effective Java》
java·开发语言
麦聪聊数据1 天前
为何通用堡垒机无法在数据库运维中实现精准风控?
数据库·sql·安全·低代码·架构
2301_790300961 天前
Python数据库操作:SQLAlchemy ORM指南
jvm·数据库·python
zfoo-framework1 天前
帧同步和状态同步
java
charlotte102410241 天前
高并发:关于在等待学校教务系统选课时的碎碎念
java·运维·网络
m0_736919101 天前
用Pandas处理时间序列数据(Time Series)
jvm·数据库·python
亓才孓1 天前
[JDBC]PreparedStatement替代Statement
java·数据库
_F_y1 天前
C++重点知识总结
java·jvm·c++
打工的小王1 天前
Spring Boot(三)Spring Boot整合SpringMVC
java·spring boot·后端