PHP乐观锁结合事务扣除余额失败:如何保证并发情况下余额正确扣除?

wufei123 2025-04-06 阅读:29 评论:0
PHP乐观锁与数据库事务结合扣除余额:问题分析与解决方案 本文探讨在PHP环境下,使用乐观锁和数据库事务进行余额扣除时,如何避免并发问题导致余额扣除失败或数据不一致的情况。 我们将分析错误代码,并提供正确的解决方案。 问题代码分析及错误...

php乐观锁结合事务扣除余额失败:如何保证并发情况下余额正确扣除?

PHP乐观锁与数据库事务结合扣除余额:问题分析与解决方案

本文探讨在PHP环境下,使用乐观锁和数据库事务进行余额扣除时,如何避免并发问题导致余额扣除失败或数据不一致的情况。 我们将分析错误代码,并提供正确的解决方案。

问题代码分析及错误原因:

以下代码片段试图通过乐观锁和事务保证并发环境下余额扣除的正确性,但存在缺陷:

错误代码片段一:

PHP
public function userbuy()
{
    $user = $this->getuser(); 
    $oldmoney = $user['balance']; 
    $orderoffer = $this->getordermoney($orderid); 
    if($oldmoney < $orderoffer['price']) $this->error('账户余额不足');

    //乐观锁方案 (错误之处)
    $newmoney = $oldmoney - $orderoffer['price']; 
    $newuser = smsuser::where(['id' => $user['id'],'balance' => $oldmoney])->find();
    if(!$newuser) $this->error('用户不存在');
    //开启数据库事务
    db::transaction(function () use($newuser,$orderid,$newmoney){
        $newuser->balance = $newmoney;
        $result = $newuser->save();
        if(!$result) $this->error('保存余额失败');

         //创建订单 code
         //扣除库存 code
         //创建用户余额变动记录 code    
        db::commit(); //提交事务 (多余操作)        
    });
}

错误原因:

  1. 乐观锁位置错误: find() 方法在事务外部执行。多个并发请求都获取到相同的 $oldmoney 值。 只有第一个请求的 save() 操作成功,后续请求由于 balance 不再等于 $oldmoney 而失败。 乐观锁的条件判断应该在 update 语句中进行。

  2. db::commit() 多余: db::transaction() 方法本身会自动提交事务,除非发生异常回滚。手动调用 db::commit() 冗余且可能掩盖错误。

错误代码片段二:

PHP
public function userbuy()
{
    // ... (代码同上,省略部分) ...

    //乐观锁方案 (错误之处)
    $newUser->balance = $newMoney;
    $result = $newUser->save();
    if(!$result) $this->error('保存余额失败');

    //开启数据库事务 (错误之处)
    Db::transaction(function () use(){
         //创建订单 code
         //扣除库存 code
         //创建用户余额变动记录 code    
        Db::commit(); //提交事务 (多余操作)         
    });
}

错误原因:

余额更新操作在事务外部执行。如果后续操作(创建订单、扣除库存等)失败,余额已经更新,导致数据不一致。

正确解决方案:

应该将乐观锁的条件判断放在数据库 update 语句中,并将所有需要保证原子性的操作都包含在同一个事务中。

正确代码:

PHP
public function userbuy()
{
    $user = $this->getuser();
    $oldmoney = $user['balance'];
    $orderoffer = $this->getordermoney($orderid);
    if ($oldmoney < $orderoffer['price']) $this->error('账户余额不足');

    $newmoney = $oldmoney - $orderoffer['price'];

    // 使用数据库的乐观锁机制 (例如MySQL)
    $affectedRows = smsuser::where('id', $user['id'])
        ->where('balance', $oldmoney)
        ->update(['balance' => $newmoney]);

    if ($affectedRows === 0) {
        $this->error('余额更新失败,可能存在并发冲突'); // 乐观锁失败
        return;
    }

    // 开启事务,包含所有后续操作
    db::transaction(function () use ($orderid, $newmoney, $user) {
        // 创建订单 code
        // 扣除库存 code
        // 创建用户余额变动记录 code  (确保这些操作也包含在事务中)
    });
}

这个解决方案将余额更新和后续操作都放在同一个事务中,并使用数据库提供的乐观锁机制来保证数据一致性。如果任何操作失败,事务会自动回滚,确保数据安全。 避免了手动提交事务,简化了代码并提高了可读性。 如果 update 语句影响的行数为0,则表示乐观锁失败,需要处理并发冲突。 可以使用重试机制或其他策略处理此类情况。 具体的乐观锁实现方式取决于使用的数据库系统。

以上就是PHP乐观锁结合事务扣除余额失败:如何保证并发情况下余额正确扣除?的详细内容,更多请关注知识资源分享宝库其它相关文章!

版权声明

本站内容来源于互联网搬运,
仅限用于小范围内传播学习,请在下载后24小时内删除,
如果有侵权内容、不妥之处,请第一时间联系我们删除。敬请谅解!
E-mail:dpw1001@163.com

分享:

扫一扫在手机阅读、分享本文

发表评论
热门文章
  • BioWare埃德蒙顿工作室面临关闭危机,龙腾世纪制作总监辞职引关注(龙腾.总监.辞职.危机.面临.....)

    BioWare埃德蒙顿工作室面临关闭危机,龙腾世纪制作总监辞职引关注(龙腾.总监.辞职.危机.面临.....)
    知名变性人制作总监corrine busche离职bioware,引发业界震荡!外媒“smash jt”独家报道称,《龙腾世纪:影幢守护者》制作总监corrine busche已离开bioware,此举不仅引发了关于个人职业发展方向的讨论,更因其可能预示着bioware埃德蒙顿工作室即将关闭而备受关注。本文将深入分析busche离职的原因及其对bioware及游戏行业的影响。 Busche的告别信:挑战与感激并存 据“Smash JT”获得的内部邮件显示,Busche离职原...
  • 闪耀暖暖靡城永恒怎么样-闪耀暖暖靡城永恒套装介绍(闪耀.暖暖.套装.介绍.....)

    闪耀暖暖靡城永恒怎么样-闪耀暖暖靡城永恒套装介绍(闪耀.暖暖.套装.介绍.....)
    闪耀暖暖钻石竞技场第十七赛季“华梦泡影”即将开启!全新闪耀性感套装【靡城永恒】震撼来袭!想知道如何获得这套精美套装吗?快来看看吧! 【靡城永恒】套装设计理念抢先看: 设计灵感源于夜色中的孤星,象征着淡然、漠视一切的灰色瞳眸。设计师希望通过这套服装,展现出在虚幻与真实交织的夜幕下,一种独特的魅力。 服装细节考究,从面料的光泽、鞋跟声响到裙摆的弧度,都力求完美还原设计初衷。 【靡城永恒】套装设计亮点: 闪耀的绸缎与金丝交织,轻盈的羽毛增添华贵感。 这套服装仿佛是从无尽的黑...
  • 蛋仔派对2025最新皮肤兑换码汇总 最新皮肤兑换码一览(兑换.皮肤.最新.派对.汇总.....)

    蛋仔派对2025最新皮肤兑换码汇总 最新皮肤兑换码一览(兑换.皮肤.最新.派对.汇总.....)
    蛋仔派对2025最新皮肤兑换码大放送!游戏内新增多款皮肤兑换码,包含最新、福利和通用三种类型,助你轻松获取精美奖励! 赶紧来看看如何兑换吧! 兑换码列表: 最新兑换码: ccewndj4k4k、cdkqdfm4fh、peetnmp4ef、cdxymk8f67 福利兑换码: cca863ywtfa、eggy2310am、eggy2311gz、eggyeggy9wz 通用兑换码: pec74dkcty、jsrqkrrjmh、cd3wt7wrph、ccepn7d8cjf...
  • python怎么调用其他文件函数

    python怎么调用其他文件函数
    在 python 中调用其他文件中的函数,有两种方式:1. 使用 import 语句导入模块,然后调用 [模块名].[函数名]();2. 使用 from ... import 语句从模块导入特定函数,然后调用 [函数名]()。 如何在 Python 中调用其他文件中的函数 在 Python 中,您可以通过以下两种方式调用其他文件中的函数: 1. 使用 import 语句 优点:简单且易于使用。 缺点:会将整个模块导入到当前作用域中,可能会导致命名空间混乱。 步骤:...
  • 俄罗斯引擎yandex入口官网地址 yandex网址在线免费进入(俄罗斯.官网.在线免费.入口.地址......)

    俄罗斯引擎yandex入口官网地址 yandex网址在线免费进入(俄罗斯.官网.在线免费.入口.地址......)
    俄罗斯引擎yandex官网地址入口在哪里?这是不少网友都关注的问题,接下来由php小编为大家带来yandex网址在线免费进入,感兴趣的网友一起随小编来瞧瞧吧! 俄罗斯引擎yandex入口官网地址 1、俄罗斯引擎yandex入口官网地址☜☜☜☜☜点击进入 2、yandex网址在线免费进入☜☜☜☜☜点击进入 【俄罗斯引擎yandex】 1、Yandex的搜索引擎在俄罗斯拥有极高的市场份额,其算法针对俄语和斯拉夫语系进行了优化,能更好地理解用户意图,提供更精准的搜索结果。它不仅...