1,安裝redis,根據自己的php版本安裝對應的redis擴展(此步驟簡單的描述一下)
1.1,安裝 php_igbinary.dll,php_redis.dll擴展此處需要注意你的php版本如圖:
1.2,php.ini文件新增 extension=php_igbinary.dll;extension=php_redis.dll兩處擴展
ok此處已經完成第一步redis環境搭建完成看看phpinfo
項目中實際使用redis
2.1,第一步配置redis參數如下,redis安裝的默認端口為6379:
2.2,實際函數中使用redis:
1234567891011 | /** * redis連接 * @access private * @return resource * @author bieanju */ private function connectRedis(){ $redis = new /Redis(); $redis ->connect(C( "REDIS_HOST" ),C( "REDIS_PORT" )); return $redis ; } |
2.3,秒殺的核心問題是在大并發的情況下不會超出庫存的購買,這個就是處理的關鍵所以思路是第一步在秒殺類的先做一些基礎的數據生成:
1234567891011 | //現在初始化里面定義后邊要使用的redis參數 public function _initialize(){ parent::_initialize(); $goods_id = I( "goods_id" , '0' , 'intval' ); if ( $goods_id ){ $this ->goods_id = $goods_id ; $this ->user_queue_key = "goods_" . $goods_id . "_user" ; //當前商品隊列的用戶情況 $this ->goods_number_key = "goods" . $goods_id ; //當前商品的庫存隊列 } $this ->user_id = $this ->user_id ? $this ->user_id : $_session [ 'uid' ]; } |
2.4,第二步就是關鍵所在,用戶在進入商品詳情頁前先將當前商品的庫存進行隊列存入redis如下:
123456789101112131415161718192021222324252627282930 | /** * 訪問產品前先將當前產品庫存隊列 * @access public * @author bieanju */ public function _before_detail(){ $where [ 'goods_id' ] = $this ->goods_id; $where [ 'start_time' ] = array ( "lt" ,time()); $where [ 'end_time' ] = array ( "gt" ,time()); $goods = M( "goods" )->where( $where )->field( 'goods_num,start_time,end_time' )->find(); ! $goods && $this ->error( "當前秒殺已結束!" ); if ( $goods [ 'goods_num' ] > $goods [ 'order_num' ]){ $redis = $this ->connectRedis(); $getUserRedis = $redis ->hGetAll( "{$this->user_queue_key}" ); $gnRedis = $redis ->llen( "{$this->goods_number_key}" ); /* 如果沒有會員進來隊列庫存 */ if (! count ( $getUserRedis ) && ! $gnRedis ){ for ( $i = 0; $i < $goods [ 'goods_num' ]; $i ++) { $redis ->lpush( "{$this->goods_number_key}" , 1); } } $resetRedis = $redis ->llen( "{$this->goods_number_key}" ); if (! $resetRedis ){ $this ->error( "系統繁忙,請稍后搶購!" ); } } else { $this ->error( "當前產品已經秒殺完!" ); } } |
接下來要做的就是用Ajax來異步的處理用戶點擊購買按鈕進行符合條件的數據進入購買的排隊隊列(如果當前用戶沒在當前產品用戶的隊列就進入排隊并且pop一個庫存隊列,如果在就拋出,):
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849 | /** * 搶購商品前處理當前會員是否進入隊列 * @access public * @author bieanju */ public function goods_number_queue(){ ! $this ->user_id && $this ->ajaxReturn( array ( "status" => "-1" , "msg" => "請先登錄" )); $model = M( "Flash_sale" ); $where [ 'goods_id' ] = $this ->goods_id; $goods_info = $model ->where( $where )->find(); ! $goods_info && $this ->error( "對不起當前商品不存在或已下架!" ); /* redis 隊列 */ $redis = $this ->connectRedis(); /* 進入隊列 */ $goods_number_key = $redis ->llen( "{$this->goods_number_key}" ); if (! $redis ->hGet( "{$this->user_queue_key}" , $this ->user_id)) { $goods_number_key = $redis ->lpop( "{$this->goods_number_key}" ); } if ( $goods_number_key ){ // 判斷用戶是否已在隊列 if (! $redis ->hGet( "{$this->user_queue_key}" , $this ->user_id)) { // 插入搶購用戶信息 $userinfo = array ( "user_id" => $this ->user_id, "create_time" => time() ); $redis ->hSet( "{$this->user_queue_key}" , $this ->user_id, serialize( $userinfo )); $this ->ajaxReturn( array ( "status" => "1" )); } else { $modelCart = M( "cart" ); $condition [ 'user_id' ] = $this ->user_id; $condition [ 'goods_id' ] = $this ->goods_id; $condition [ 'prom_type' ] = 1; $cartlist = $modelCart ->where( $condition )-> count (); if ( $cartlist > 0){ $this ->ajaxReturn( array ( "status" => "2" )); } else { $this ->ajaxReturn( array ( "status" => "1" )); } } } else { $this ->ajaxReturn( array ( "status" => "-1" , "msg" => "系統繁忙,請重試!" )); } } |
附加一個調試的函數,刪除指定隊列值:
1234567891011121314 | public function clearRedis(){ set_time_limit(0); $redis = $this ->connectRedis(); //$Rd = $redis->del("{$this->user_queue_key}"); $Rd = $redis ->hDel( "goods49" , '用戶id' '); $a = $redis ->hGet( "goods_49_user" , '用戶id' ); if (! $a ){ dump( $a ); } if ( $Rd == 0){ exit ( "Redis隊列已釋放!" ); } } |
新聞熱點
疑難解答
圖片精選