前段时间面试的时候,一直被问到如何设计一个秒杀活动,可是无奈没有此方面的实际经验,因此只好凭着本身的理解和一些资料去设计这么一个程序
主要利用到了redis的string和set,string主要是利用它的k-v结构去对库存进行处理,也能够用list的数据结构来处理商品的库存,set则用来确保用户进行重复的提交
其中咱们最主要解决的问题是
-防止并发产生超抢/超卖php
class MiaoSha{ const MSG_REPEAT_USER = '请勿重复参与'; const MSG_EMPTY_STOCK = '库存不足'; const MSG_KEY_NOT_EXIST = 'key不存在'; const IP_POOL = 'ip_pool'; const USER_POOL = 'user_pool'; /** @var Redis */ public $redis; public $key; public function __construct($key = '') { $this->checkKey($key); $this->redis = new Redis(); //todo 链接池 $this->redis->connect('127.0.0.1'); } public function checkKey($key = '') { if(!$key) { throw new Exception(self::MSG_KEY_NOT_EXIST); } else { $this->key = $key; } } public function setStock($value = 0) { if($this->redis->exists($this->key) == 0) { $this->redis->set($this->key,$value); } } public function checkIp($ip = 0) { $sKey = $this->key . self::IP_POOL; if(!$ip || $this->redis->sIsMember($sKey,$ip)) { throw new Exception(self::MSG_REPEAT_USER); } } public function checkUser($user = 0) { $sKey = $this->key . self::USER_POOL; if(!$user || $this->redis->sIsMember($sKey,$user)) { throw new Exception(self::MSG_REPEAT_USER); } } public function checkStock($user = 0, $ip = 0) { $num = $this->redis->decr($this->key); if($num < 0 ) { throw new Exception(self::MSG_EMPTY_STOCK); } else { $this->redis->sAdd($this->key . self::USER_POOL, $user); $this->redis->sAdd($this->key . self::IP_POOL, $ip); //todo add to mysql echo 'success' . PHP_EOL; error_log('success' . $user . PHP_EOL,3,'/var/www/html/demo/log/debug.log'); } } /** * @note:此种作法不能防止并发 * @func checkStockFail * @param int $user * @param int $ip * @throws Exception */ public function checkStockFail($user = 0,$ip = 0) { $num = $this->redis->get($this->key); if($num > 0 ){ $this->redis->sAdd($this->key . self::USER_POOL, $user); $this->redis->sAdd($this->key . self::IP_POOL, $ip); //todo add to mysql echo 'success' . PHP_EOL; error_log('success' . $user . PHP_EOL,3,'/var/www/html/demo/log/debug.log'); $num--; $this->redis->set($this->key,$num); } else { throw new Exception(self::MSG_EMPTY_STOCK); } } }
function test() { try{ $key = 'cup_'; $handler = new MiaoSha($key); $handler->setStock(10); $user = rand(1,10000); $ip = $user; $handler->checkIp($ip); $handler->checkUser($user); $handler->checkStock($user,$ip); } catch (\Exception $e) { echo $e->getMessage() . PHP_EOL; error_log('fail' . $e->getMessage() .PHP_EOL,3,'/var/www/html/demo/log/debug.log'); } } function test2() { try{ $key = 'cup_'; $handler = new MiaoSha($key); $handler->setStock(10); $user = rand(1,10000); $ip = $user; $handler->checkIp($ip); $handler->checkUser($user); $handler->checkStockFail($user,$ip); //不能防止并发的 } catch (\Exception $e) { echo $e->getMessage() . PHP_EOL; error_log('fail' . $e->getMessage() .PHP_EOL,3,'/var/www/html/demo/log/debug.log'); } }
测试环境说明html
在服务端代码里面咱们有两个函数分别是checkStock和checkStockFail,其中checkStockFail不能在高并发的状况下效果不好,不能在redis层面保证库存为0的时候终止操做。
咱们利用ab工具进行测试
其中www.hello.com
是配置的虚拟主机名称 flash-sale.php
是咱们脚本的名称mysql
#第1种状况 500并发下 用客户端的test2()去执行 ab -n 500 -c 100 www.hello.com/flash-sale.php
log日志的记录结果:面试
#第2种状况 5000并发下 用客户端的test2()去执行 ab -n 5000 -c 1000 www.hello.com/flash-sale.php
log日志的记录结果:redis
#第3种状况 500并发下 用客户端的test()去执行 ab -n 500 -c 100 www.hello.com/flash-sale.php
log日志的记录结果:sql
#第4种状况 5000并发下 用客户端的test()去执行 ab -n 5000 -c 1000 www.hello.com/flash-sale.php
log日志的记录结果:api
咱们从日志中能够很明显的看出第三、4中状况下,能够保证商品的数量老是咱们设置的库存值10,可是在状况一、2下,则产生了超卖的现象
redis来控制并发主要是利用了其api都是原子性操做的优点,从checkStock和checkStockFail中能够看出,一个是直接decr对库存进行减一操做,因此不存在并发的状况,可是另外一个方法是将库存值先取出作减一操做而后再从新赋值,这样的话,在并发下,多个进程会读取到多个库存为1的值,所以会产生超卖的状况数据结构