今回は、php+redisメッセージキューを使って急ぎ買いを実装する手順(コード付き)を詳しく解説します。php+redisメッセージキューを導入する際に注意すべき注意点とは何ですか。以下は実際的なケースですので、一緒に見てみましょう。
実装機能:
1. 同時実行の過剰販売を防ぐための Redis キューに基づく
2. 同時実行の過剰販売を防止するための mysql トランザクションと排他的ロックに基づく
Redis キューのワークフローに基づく:
1. スタッフが商品テーブルの在庫に基づいて Redis 製品在庫キューを作成します
2. クライアントはフラッシュセール API にアクセスします
3. Web サーバーは最初に Redis 製品在庫キューの残りの在庫キーの内容をクエリします
4. 在庫が残っている場合redis キューにある場合は、mysql で注文を作成し、在庫を削除し、スナップアップに成功します
5. Redis キューに残りがない場合は、在庫が不十分でスナップアップに失敗したことを示すメッセージが表示されます
。 mysql トランザクションと排他的ロックのワークフローに基づく:
1. トランザクションを開く
2. インベントリをクエリし、暗黙的に書き込みロック (排他的ロック) を設定する: SELECT * FROM Goods WHERE id = 1 FOR UPDATE
3. 注文を生成する
4.在庫、書き込みロック (排他的ロック) を暗黙的に設定します: UPDATE Goods SET counts = counts – 1 WHERE id = 1
5. commit、ロックを解放します
注: 2 番目のステップで共有ロックを設定できます。そうしないと、共有ロックが発生する可能性があります。行き詰まり。
コード:
<?php /********************************************** * 抢购模块 * * @author liubin * @date 2016-02-10 * * ab -n 1000 -c 100 http://192.168.16.73/Seckill/buy.php * */ class seckill extends common { private $_orderModel = null; private $_goodsModel = null; private $_redis = null; /* * 错误信息 */ protected $_error = ''; /** * 构造器 * */ public function construct() { if($this->_orderModel === null){ $this->_orderModel = new OrderModel(); } if($this->_goodsModel === null){ $this->_goodsModel = new GoodsModel(); } if($this->_redis === null){ $this->_redis = new QRedis(); } } /* * 秒杀API * * @author liubin * @date 2017-02-10 */ public function addQsec(){ $gid = intval($_GET['gid']); $type = isset($_GET['type']) ? $_GET['type'] : 'mysql'; switch ($type) { case 'mysql': $this->order_check_mysql($gid); echo $this->getError(); break; case 'redis': $this->order_check_redis($gid); echo $this->getError(); break; case 'transaction': $this->order_check_transaction($gid); echo $this->getError(); break; default: echo '类型错误'; break; } } /* * 获取错误信息 * * @author liubin * @date 2017-02-10 */ public function getError(){ return $this->_error; } /* * 基于mysql验证库存信息 * @desc 高并发下会导致超卖 * * @author liubin * @date 2017-02-10 */ protected function order_check_mysql($gid){ $model = $this->_goodsModel; $pdo = $model->getHandler(); $gid = intval($gid); /* * 1:$sql_forlock如果不加事务,不加写锁: * 超卖非常严重,就不说了 * * 2:$sql_forlock如果不加事务,只加写锁: * 第一个会话读$sql_forlock时加写锁,第一个会话$sql_forlock查询结束会释放该行锁. * 第二个会话在第一个会话释放后读$sql_forlock的写锁时,会再次$sql_forlock查库存 * 导致超卖现象产生 * */ $sql_forlock = 'select * from goods where id = '.$gid .' limit 1 for update'; //$sql_forlock = 'select * from goods where id = '.$gid .' limit 1'; $result = $pdo->query($sql_forlock,PDO::FETCH_ASSOC); $goodsInfo = $result->fetch(); if($goodsInfo['counts']>0){ //去库存 $gid = $goodsInfo['id']; $sql_inventory = 'UPDATE goods SET counts = counts - 1 WHERE id = '.$gid; $result = $this->_goodsModel->exect($sql_inventory); if($result){ //创订单 $data = []; $data['order_id'] = $this->_orderModel->buildOrderNo(); $data['goods_id'] = $goodsInfo['id']; $data['addtime'] = time(); $data['uid'] = 1; $order_rs = $this->_orderModel->create_order($data); if($order_rs){ $this->_error = '购买成功'; return true; } } } $this->_error = '库存不足'; return false; } /* * 基于redis队列验证库存信息 * @desc Redis是底层是单线程的,命令执行是原子操作,包括lpush,lpop等.高并发下不会导致超卖 * * @author liubin * @date 2017-02-10 */ protected function order_check_redis($gid){ $goodsInfo = $this->_goodsModel->getGoods($gid); if(!$goodsInfo){ $this->_error = '商品不存在'; return false; } $key = 'goods_list_'.$goodsInfo['id']; $count = $this->_redis->getHandel()->lpop($key); if(!$count){ $this->_error = '库存不足'; return false; } //生成订单 $data = []; $data['order_id'] = $this->_orderModel->buildOrderNo(); $data['goods_id'] = $goodsInfo['id']; $data['addtime'] = time(); $data['uid'] = 1; $order_rs = $this->_orderModel->create_order($data); //库存减少 $gid = $goodsInfo['id']; $sql = 'UPDATE goods SET counts = counts - 1 WHERE id = '.$gid; $result = $this->_goodsModel->exect($sql); $this->_error = '购买成功'; return true; } /* * 基于mysql事务验证库存信息 * @desc 事务 和 行锁 模式,高并发下不会导致超卖,但效率会慢点 * @author liubin * @date 2017-02-10 说明: 如果$sql_forlock不加写锁,并发时,$sql_forlock查询的记录存都大于0,可以减库存操作. 如果$sql_forlock加了写锁,并发时,$sql_forlock查询是等待第一次链接释放后查询.所以库存最多就是5 */ protected function order_check_transaction($gid){ $model = $this->_goodsModel; $pdo = $model->getHandler(); $gid = intval($gid); try{ $pdo->beginTransaction();//开启事务处理 /* * 1:$sql_forlock如果只加事务,不加写锁: * 开启事务 * 因为没有加锁,读$sql_forlock后,并发时$sql_inventory之前还可以再读。 * $sql_inventory之后和commit之前才会锁定 * 出现超卖跟事务的一致性不冲突 * * * 2:$sql_forlock如果加了事务,又加读锁: * 开启事务 * 第一个会话读$sql_forlock时加读锁,并发时,第二个会话也允许获得$sql_forlock的读锁, * 但是在第一个会话执行去库存操作时(写锁),写锁便会等待第二个会话的读锁,第二个会话执行写操作时,写锁便会等待第一个会话的读锁, * 出现死锁 * 3:$sql_forlock如果加了事务,又加写锁: * 开启事务 * 第一个会话读$sql_forlock时加写锁,直到commit才会释放写锁,并发查询不会出现超卖现象。 * */ $sql_forlock = 'select * from goods where id = '.$gid .' limit 1 for update'; //$sql_forlock = 'select * from goods where id = '.$gid .' limit 1 LOCK IN SHARE MODE'; //$sql_forlock = 'select * from goods where id = '.$gid .' limit 1'; $result = $pdo->query($sql_forlock,PDO::FETCH_ASSOC); $goodsInfo = $result->fetch(); if($goodsInfo['counts']>0){ //去库存 $gid = $goodsInfo['id']; $sql_inventory = 'UPDATE goods SET counts = counts - 1 WHERE id = '.$gid; $result = $this->_goodsModel->exect($sql_inventory); if(!$result){ $pdo->rollBack(); $this->_error = '库存减少失败'; return false; } //创订单 $data = []; $data['id'] = 'null'; $data['order_id'] = $this->_orderModel->buildOrderNo(); $data['goods_id'] = $goodsInfo['id']; $data['uid'] = 'abc'; $data['addtime'] = time(); $sql = 'insert into orders (id,order_id,goods_id,uid,addtime) values ('.$data['id'].',"'.$data['order_id'].'","'.$data['goods_id'].'","'.$data['uid'].'","'.$data['addtime'].'")'; $result = $pdo->exec($sql); if(!$result){ $pdo->rollBack(); $this->_error = '订单创建失败'; return false; } $pdo->commit();//提交 $this->_error = '购买成功'; return true; }else{ $this->_error = '库存不足'; return false; } }catch(PDOException $e){ echo $e->getMessage(); $pdo->rollBack(); } } /* * 创建订单 * mysql 事物处理,也可以用存储过程 * */ private function create_order($goodsInfo){ //生成订单 $data = []; $data['order_id'] = $this->_orderModel->buildOrderNo(); $data['goods_id'] = $goodsInfo['id']; $data['addtime'] = time(); $data['uid'] = 1; $order_rs = $this->_orderModel->create_order($data); //库存减少 $gid = $goodsInfo['id']; $sql = 'UPDATE goods SET counts = counts - 1 WHERE id = '.$gid; $result = $this->_goodsModel->exect($sql); return true; } }
この記事の事例を読んだ後は、この方法を習得したと思います。さらに興味深い情報については、php 中国語 Web サイトの他の関連記事に注目してください。
推奨読書:
PHP が緊急購入の高同時リクエストを処理する方法の詳細な説明
Windows でリアルタイムの CPU メモリ使用量を取得する方法
以上がphp+redisメッセージキューを使った駆け込み購入の実装手順を詳しく解説(コード付き)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

方法:1、用“str_replace(" ","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\ \;||\xc2\xa0)/","其他字符",$str)”语句。

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

在php中,可以使用substr()函数来读取字符串后几个字符,只需要将该函数的第二个参数设置为负值,第三个参数省略即可;语法为“substr(字符串,-n)”,表示读取从字符串结尾处向前数第n个字符开始,直到字符串结尾的全部字符。


ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

AI Hentai Generator
AIヘンタイを無料で生成します。

人気の記事

ホットツール

DVWA
Damn Vulnerable Web App (DVWA) は、非常に脆弱な PHP/MySQL Web アプリケーションです。その主な目的は、セキュリティ専門家が法的環境でスキルとツールをテストするのに役立ち、Web 開発者が Web アプリケーションを保護するプロセスをより深く理解できるようにし、教師/生徒が教室環境で Web アプリケーションを教え/学習できるようにすることです。安全。 DVWA の目標は、シンプルでわかりやすいインターフェイスを通じて、さまざまな難易度で最も一般的な Web 脆弱性のいくつかを実践することです。このソフトウェアは、

PhpStorm Mac バージョン
最新(2018.2.1)のプロフェッショナル向けPHP統合開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

MinGW - Minimalist GNU for Windows
このプロジェクトは osdn.net/projects/mingw に移行中です。引き続きそこでフォローしていただけます。 MinGW: GNU Compiler Collection (GCC) のネイティブ Windows ポートであり、ネイティブ Windows アプリケーションを構築するための自由に配布可能なインポート ライブラリとヘッダー ファイルであり、C99 機能をサポートする MSVC ランタイムの拡張機能が含まれています。すべての MinGW ソフトウェアは 64 ビット Windows プラットフォームで実行できます。

ZendStudio 13.5.1 Mac
強力な PHP 統合開発環境
