mysql - PHP高并发下单用事务可以解决吗?

查看:278
本文介绍了mysql - PHP高并发下单用事务可以解决吗?的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

问 题

一个下单的小示例(上代码,没加事务的时候):

class IndexController extends Controller {

    public function index(){
        $stock = M('stock');
        $log = M('log');
        $condition['id'] = 1;
        if($stock->where($condition)->getField('stock_left') > 0) {
            $stock->where($condition)->setDec("stock_left");
            $data['op'] = 1;
            $log->add($data);
        } else {
            echo "已经没剩余了";
        }
    }
}

库存默认有100个:

日志表:

Apache ab工具并发一下:
ab -n 1200 -c 1200 -w http://localhost/queue/index.php >> D:/1.html

结果出并发问题(很自然的):

然后加了事务控制之后:

class IndexController extends Controller {

    public function index(){
        $stock = M('stock');
        $log = M('log');
        $condition['id'] = 1;
        M()->startTrans();
        if($stock->where($condition)->getField('stock_left') > 0) {
            $res1 = $stock->where($condition)->setDec("stock_left");
            $data['op'] = 1;
            $res2 = $log->add($data);
            if($res1 !== false && $res2) {
                M()->commit();
            } else {
                M()->rollback();
            }
        } else {
            echo "已经没剩余了";
        }
    }
}

再测试并发一下:
ab -n 1200 -c 1200 -w http://localhost/queue/index.php >> D:/1.html
结果呢,输出的结果没问题(但是真的解决了并发问题吗?):

很多人说了用Redis队列来做,具体实施我还是有点不太清楚,请大家帮忙

解决方案

事务跟并发没有任何关系。你使用事务只能保证这一段逻辑成功或者失败,而不能保证并发时能控制住你的程序逻辑。
对并发进行控制还是需要锁 来解决,比如楼上有提到的mysql 实现乐观锁。
如 upadte table set a = a - 1 where a = 5; 只有在a=5的情况下这个update才会真正修改数据,使用这种方法是可以保证如果要修改的数据版本跟你预想中的不同,就不进行操作,通过影响行数来判断是否有修改,然后继续下面的操作或者退出。

还有就是使用排他锁,如果单机可以直接使用flock 来达到阻塞锁的目的。
或者redis和memcache来实现锁。
请求比较多的的情况下推荐使用redis,memcache来进行锁操作,或者考虑用消息队列来处理并发的情况。

这篇关于mysql - PHP高并发下单用事务可以解决吗?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆