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

相关推荐
XDHCOM17 小时前
ORA-32484重复列名错误,ORACLE数据库CYCLE子句故障修复与远程处理方案
数据库·oracle
云烟成雨TD17 小时前
Spring AI Alibaba 1.x 系列【6】ReactAgent 同步执行 & 流式执行
java·人工智能·spring
于慨17 小时前
Lambda 表达式、方法引用(Method Reference)语法
java·前端·servlet
swg32132117 小时前
Spring Boot 3.X Oauth2 认证服务与资源服务
java·spring boot·后端
翻斗包菜17 小时前
PostgreSQL 日常维护完全指南:从基础操作到高级运维
运维·数据库·postgresql
gelald17 小时前
SpringBoot - 自动配置原理
java·spring boot·后端
殷紫川17 小时前
深入理解 AQS:从架构到实现,解锁 Java 并发编程的核心密钥
java
呆瑜nuage17 小时前
MySQL表约束详解:8大核心约束实战指南
数据库·mysql
一轮弯弯的明月17 小时前
贝尔数求集合划分方案总数
java·笔记·蓝桥杯·学习心得
一博一言18 小时前
Oracle高版本Version_Count问题处理排查
oracle·dba