>백엔드 개발 >PHP 튜토리얼 >PHP+redis는 스냅업 기능을 실현합니다.

PHP+redis는 스냅업 기능을 실현합니다.

小云云
小云云원래의
2018-02-09 09:21:003381검색

앞서 쇼핑몰 플래시 세일 기능을 구현하기 위한 php와 redis의 코드 공유를 공유했습니다. 이 글에서는 주로 php+redis 메시지 대기열 러시 구매 구현 코드를 자세히 소개했습니다. 관심 있는 친구들은 참고할 수 있습니다. . 모두에게 도움이 되기를 바랍니다.

구현 기능:

1. 높은 동시성 과잉 판매를 방지하기 위한 redis 대기열 기반
2. 높은 동시성 과잉 판매를 방지하기 위한 mysql 트랜잭션 및 독점 잠금 기반

redis 대기열 워크플로 기반:

1 . 상품 테이블의 재고를 기반으로 한 Redis 제품 재고 대기열
2. 클라이언트가 플래시 판매 API에 액세스합니다
3. 웹 서버는 먼저 Redis 제품 재고 대기열에서 남은 재고의 주요 내용을 쿼리합니다. 그런 다음 mysql에서 주문을 생성하고 인벤토리를 제거하면 긴급 구매가 성공합니다
5. Redis 대기열에 남은 내용이 없으면 인벤토리가 부족하다는 메시지가 표시되어 긴급 구매가 실패합니다. 주요 내용

mysql 트랜잭션 및 배타적 잠금 워크플로 기반:


1. 트랜잭션 열기

2. 인벤토리를 쿼리하고 명시적으로 쓰기 잠금(배타적 잠금) 설정: SELECT * FROM products WHERE id = 1 FOR UPDATE
3. 주문 생성
4. 재고 제거 및 암시적으로 쓰기 잠금 설정(독점 잠금): UPDATE 상품 SET counts = counts – 1 WHERE id = 1
5. 커밋, 잠금 해제

참고: 설정할 수 있습니다. 두 번째 단계에서 공유 잠금을 해제합니다. 그렇지 않으면 교착 상태가 발생할 수 있습니다.

코드:


<?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 = &#39;&#39;;
 /**
  * 构造器
  *
 */
 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[&#39;gid&#39;]);
  $type = isset($_GET[&#39;type&#39;]) ? $_GET[&#39;type&#39;] : &#39;mysql&#39;;
  switch ($type) {
   case &#39;mysql&#39;:
    $this->order_check_mysql($gid);
    echo $this->getError();
    break;
   case &#39;redis&#39;:
    $this->order_check_redis($gid);
    echo $this->getError();
    break;
   case &#39;transaction&#39;:
    $this->order_check_transaction($gid);
    echo $this->getError();
    break;
   default:
    echo &#39;类型错误&#39;;
    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 = &#39;select * from goods where id = &#39;.$gid .&#39; limit 1 for update&#39;;
  //$sql_forlock = &#39;select * from goods where id = &#39;.$gid .&#39; limit 1&#39;;
  $result = $pdo->query($sql_forlock,PDO::FETCH_ASSOC);
  $goodsInfo = $result->fetch();

  if($goodsInfo[&#39;counts&#39;]>0){

   //去库存
   $gid = $goodsInfo[&#39;id&#39;];
   $sql_inventory = &#39;UPDATE goods SET counts = counts - 1 WHERE id = &#39;.$gid;
   $result = $this->_goodsModel->exect($sql_inventory);
   if($result){
    //创订单
    $data    = [];
    $data[&#39;order_id&#39;] = $this->_orderModel->buildOrderNo();
    $data[&#39;goods_id&#39;] = $goodsInfo[&#39;id&#39;];
    $data[&#39;addtime&#39;] = time();
    $data[&#39;uid&#39;]  = 1;
    $order_rs = $this->_orderModel->create_order($data);
    if($order_rs){
     $this->_error = &#39;购买成功&#39;;
     return true;
    }
   }
  }

  $this->_error = &#39;库存不足&#39;;
  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 = &#39;商品不存在&#39;;
   return false;
  }
  $key = &#39;goods_list_&#39;.$goodsInfo[&#39;id&#39;];
  $count = $this->_redis->getHandel()->lpop($key);
  if(!$count){
   $this->_error = &#39;库存不足&#39;;
   return false;
  }
  //生成订单
  $data    = [];
  $data[&#39;order_id&#39;] = $this->_orderModel->buildOrderNo();
  $data[&#39;goods_id&#39;] = $goodsInfo[&#39;id&#39;];
  $data[&#39;addtime&#39;] = time();
  $data[&#39;uid&#39;]  = 1;
  $order_rs = $this->_orderModel->create_order($data);

  //库存减少
  $gid = $goodsInfo[&#39;id&#39;];
  $sql = &#39;UPDATE goods SET counts = counts - 1 WHERE id = &#39;.$gid;
  $result = $this->_goodsModel->exect($sql);
  $this->_error = &#39;购买成功&#39;;
  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 = &#39;select * from goods where id = &#39;.$gid .&#39; limit 1 for update&#39;;
   //$sql_forlock = &#39;select * from goods where id = &#39;.$gid .&#39; limit 1 LOCK IN SHARE MODE&#39;;
   //$sql_forlock = &#39;select * from goods where id = &#39;.$gid .&#39; limit 1&#39;;
   $result = $pdo->query($sql_forlock,PDO::FETCH_ASSOC);
   $goodsInfo = $result->fetch();

   if($goodsInfo[&#39;counts&#39;]>0){

    //去库存
    $gid = $goodsInfo[&#39;id&#39;];
    $sql_inventory = &#39;UPDATE goods SET counts = counts - 1 WHERE id = &#39;.$gid;
    $result = $this->_goodsModel->exect($sql_inventory);

    if(!$result){
     $pdo->rollBack();
     $this->_error = &#39;库存减少失败&#39;;
     return false;
    }

    //创订单
    $data    = [];
    $data[&#39;id&#39;]   = &#39;null&#39;;
    $data[&#39;order_id&#39;] = $this->_orderModel->buildOrderNo();
    $data[&#39;goods_id&#39;] = $goodsInfo[&#39;id&#39;];
    $data[&#39;uid&#39;]  = &#39;abc&#39;;
    $data[&#39;addtime&#39;] = time();

    $sql = &#39;insert into orders (id,order_id,goods_id,uid,addtime) values (&#39;.$data[&#39;id&#39;].&#39;,"&#39;.$data[&#39;order_id&#39;].&#39;","&#39;.$data[&#39;goods_id&#39;].&#39;","&#39;.$data[&#39;uid&#39;].&#39;","&#39;.$data[&#39;addtime&#39;].&#39;")&#39;;   
    $result = $pdo->exec($sql);
    if(!$result){
     $pdo->rollBack();
     $this->_error = &#39;订单创建失败&#39;;
     return false;
    }
    $pdo->commit();//提交
    $this->_error = &#39;购买成功&#39;;
    return true;

   }else{
    $this->_error = &#39;库存不足&#39;;
    return false;
   }
  }catch(PDOException $e){
   echo $e->getMessage();
   $pdo->rollBack();
  }


 }
 /*
  * 创建订单
  * mysql 事物处理,也可以用存储过程
  *
 */
 private function create_order($goodsInfo){
  //生成订单
  $data    = [];
  $data[&#39;order_id&#39;] = $this->_orderModel->buildOrderNo();
  $data[&#39;goods_id&#39;] = $goodsInfo[&#39;id&#39;];
  $data[&#39;addtime&#39;] = time();
  $data[&#39;uid&#39;]  = 1;
  $order_rs = $this->_orderModel->create_order($data);

  //库存减少
  $gid = $goodsInfo[&#39;id&#39;];
  $sql = &#39;UPDATE goods SET counts = counts - 1 WHERE id = &#39;.$gid;
  $result = $this->_goodsModel->exect($sql);
  return true;
 }
}

관련 권장 사항:


PHP는 잠금을 통해 동시 스냅업 기능을 구현합니다.

JS 코드를 사용하여 웹페이지 스냅업 기능을 구현하는 방법

PHP는 어떻게 처리합니까? 스냅업 기능의 높은 동시성 요청

위 내용은 PHP+redis는 스냅업 기능을 실현합니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.