《高并发秒杀抢购系统设计》PHP示例代码

一年多以前在学校分享过一次《高并发秒杀抢购系统设计》,其中有部分示例代码未能贴出,因为当时工作换电脑导致程序代码丢失,一直就没有贴出来,到编写本文时有不少朋友向我要过代码,很不好意思一直没整理就没给,近期有时间就整理了一下。时间有点久了,一些内容细节有些忘记,示例代码处理模型如有考虑不到之处,请留言给我,我会跟进测试修改,提前谢谢各位。

没有看过上一篇文章的,可以先看看一次分享《高并发秒杀抢购系统设计》

本次整理代码所用的相关程序版本:

  • PHP5.6加pthreads、redis、mysql扩展
  • Mysql5.7 ,不过用不到数据库的高级特性,5.0及以上版本支持Innodb存储引擎的就可以
  • Redis5.0.5
  • Centos7.8

尝试了PHP7.3和7.4的多线程,无论是pthreads还是parallel都出现“段错误”无法正常执行,可能和Centos环境有些关系,有能执行成功的朋友请指教一下。

准备工作,建库建表,test库就行,建表语句:

create table goods (
    id int unsigned not null auto_increment primary key,
    goodname varchar(50) not null default '',
    total int not null default 0
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

insert into goods(goodname,total) values('火车票',100);

新手错误代码

先看新手最容易犯错的代码,它的处理逻辑在单进程单线程没有并发的情况下是对的,但是在高并发下就是错误的。

error_reporting(E_ALL ^ E_DEPRECATED);
class Conf {
    public static $host = 'localhost';
    public static $port = '3306';
    public static $user = 'root';
    public static $passwd = '123';
    public static $dbname = 'test';
}
class NoLock extends Thread {
    public function run() {
        //模拟真实环境,连接数据库,每次都返回一个新的数据库连接
        $mysql = mysql_connect(Conf::$host.':'.Conf::$port,Conf::$user,Conf::$passwd,true);
        mysql_select_db(Conf::$dbname);
        //从数据库中取出库存
        $sql = "select total from goods where id=1";
        $result = mysql_query($sql,$mysql);
        $info = mysql_fetch_assoc($result);
        mysql_free_result($result);
        //获取库存余量
        $total = $info['total'];
        echo 'tid='.self::getCurrentThreadId().' total='.$total."\n";
        if($total > 0) {//判断库存是否还有
            /*
             * 这里会出现两种写法,但是结果都一样,都是错误的
             * 一种是直接数据库字段减1
             * 另一种是取出的库存数减1再写回数据库
             */
//            mysql_query("update goods set total=total-1 where id=1");
            mysql_query("update goods set total='".($total-1)."' where id=1",$mysql);
        }
        mysql_close($mysql);
    }
}

$mysql = mysql_connect(Conf::$host.':'.Conf::$port,Conf::$user,Conf::$passwd,true);
mysql_select_db(Conf::$dbname);
$sql = "update goods set total=100 where id=1";
mysql_query($sql,$mysql);
mysql_close($mysql);

$clientArr = [];
for ($i=0;$i<100;++$i) {
    $clientArr[$i] = new NoLock();
    $clientArr[$i]->start();
}
//获取结果
$mysql = mysql_connect(Conf::$host.':'.Conf::$port,Conf::$user,Conf::$passwd,true);
mysql_select_db(Conf::$dbname);
$sql = "select total from goods where id=1";
$result = mysql_query($sql,$mysql);
$info = mysql_fetch_assoc($result);
mysql_free_result($result);
mysql_close($mysql);
echo 'end total='.$info['total']."\n";

Nolock类继承PHP线程类Thread,一个线程模拟一个用户下单减库存,100个库存需要100个线程,按正常逻辑100个线程执行完毕库存是0就对。上面这段代码可直接复制到一个PHP文件,修改顶部的Mysql配置,然后多次执行(一定要多次快速执行),你能够发现好多时候最后库存大于0,有的线程读取到了相同的库存。
分析一下:100个库存,100个用户都已经完成下单,还有剩余,继续执行的话一定是要超卖了~~~

悲观锁,利用Mysql实现

error_reporting(E_ALL ^ E_DEPRECATED);
class Conf {
    public static $host = 'localhost';
    public static $port = '3306';
    public static $user = 'root';
    public static $passwd = '';
    public static $dbname = 'test';
}
//利用mysql数据库实现悲观锁
class PessimisticLock extends Thread {
    public function run() {
        //模拟真实环境,连接数据库,每次都返回一个新的数据库连接
        $mysql = mysql_connect(Conf::$host.':'.Conf::$port,Conf::$user,Conf::$passwd,true);
        mysql_select_db(Conf::$dbname);
        //从数据库中取出库存 利用Innodb更新行锁实现悲观锁
        $sql = "update goods set total=total-1 where id=1 and total>0";
        $result = mysql_query($sql,$mysql);
        //要检查修改影响的条数,执行成功但不一定修改数据
        $affectedRows = $result ? mysql_affected_rows() : 0;
        if($affectedRows) {//根据修改影响的条数进行后续操作
            echo self::getCurrentThreadId()." update ok \n";
        } else {
            echo self::getCurrentThreadId()." update err \n";
        }
        mysql_close($mysql);
    }
}

$mysql = mysql_connect(Conf::$host.':'.Conf::$port,Conf::$user,Conf::$passwd,true);
mysql_select_db(Conf::$dbname);
$sql = "update goods set total=100 where id=1";
mysql_query($sql,$mysql);
mysql_close($mysql);

$clientArr = [];
for ($i=0;$i<100;++$i) {
    $clientArr[$i] = new PessimisticLock();
    $clientArr[$i]->start();
}

//获取结果
$mysql = mysql_connect(Conf::$host.':'.Conf::$port,Conf::$user,Conf::$passwd,true);
mysql_select_db(Conf::$dbname);
$sql = "select total from goods where id=1";
$result = mysql_query($sql,$mysql);
$info = mysql_fetch_assoc($result);
mysql_free_result($result);
mysql_close($mysql);
echo 'end total='.$info['total']."\n";

这段代码多次使劲执行,最后库存都是0,所以这个方法原理上可以,也只是原理上可以,不建议直接用在高并发系统上,主要因为它会大幅度增加数据库负载。我们对系统优化一般首先着手的都是减少数据库的直接操作,因此这个方法不建议,真要用还需要看具体情况。

乐观锁,利用Redis的事务来实现

error_reporting(E_ALL ^ E_DEPRECATED);
class Conf {
    public static $host = 'localhost';
}

//利用redis事务实现乐观锁
class OptimisticLock extends Thread {
    public function run() {
        $redis = new Redis();
        $redis->connect(Conf::$host);
        do {//只要还有库存且没成功减库存就一直执行
            $goodsTotal = $redis->get('goods_total');
            echo self::getCurrentThreadId().' total='.$goodsTotal."\n";
            if($goodsTotal <= 0) break;//每次都检查是否还有库存  没有库存退出循环
            $redis->watch('goods_total');
            $redis->multi();
            $redis->decr('goods_total');
            $res = $redis->exec();
        } while(!$res);
    }
}
//初始化缓存库存数据
$redis = new Redis();
$redis->connect(Conf::$host);
$redis->set('goods_total',100);

$clientArr = [];
for ($i=0;$i<100;++$i) {
    $clientArr[$i] = new OptimisticLock();
    $clientArr[$i]->start();
}

这段代码使劲多次执行,最后库存也是0,所以也是可行的,这个方法也是首先推荐的方法,内存中的数据操作比在数据库中要快得多,负载能力会高跟多。
本分享给出的示例代码只是处理逻辑,具体应用还要根据具体服务器架构甚至是业务逻辑进行调整。有不足之处欢迎批评指正。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,458评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,454评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,171评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,062评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,440评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,661评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,906评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,609评论 0 200
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,379评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,600评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,085评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,409评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,072评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,088评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,860评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,704评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,608评论 2 270