thinkphp와 redis, queue를 결합하여 구현한 코드

不言
풀어 주다: 2023-03-30 13:32:02
원래의
3073명이 탐색했습니다.

이 글은 thinkphp+redis+queue의 구현 코드를 주로 소개하고 있는데, 편집자가 보기에는 꽤 좋다고 생각해서 지금부터 참고용으로 올려보겠습니다. 편집기를 따라 살펴보겠습니다

1. Redis를 설치하고 PHP 버전에 따라 해당 Redis 확장을 설치합니다(이 단계는 간략하게 설명됩니다). php_igbinary.dll 확장은 여기에서 주의해야 합니다. 귀하의 PHP 버전은 그림과 같습니다:

 

1.2, php.ini 파일에는 두 개의 새로운 확장자가 있습니다: Extension=php_igbinary.dll; Extension=php_redis.dll

ok Redis 환경 구축의 첫 번째 단계는 다음과 같습니다. phpinfo

이 프로젝트는 실제로 redis

2.1을 사용합니다. Redis 설치를 위한 기본 포트는 6379입니다.

<?php
/* 数据库配置 */
return array(
 &#39;DATA_CACHE_PREFIX&#39; => &#39;Redis_&#39;,//缓存前缀
 &#39;DATA_CACHE_TYPE&#39;=>&#39;Redis&#39;,//默认动态缓存为Redis
 &#39;DATA_CACHE_TIMEOUT&#39; => false,
 &#39;REDIS_RW_SEPARATE&#39; => true, //Redis读写分离 true 开启
 &#39;REDIS_HOST&#39;=>&#39;127.0.0.1&#39;, //redis服务器ip,多台用逗号隔开;读写分离开启时,第一台负责写,其它[随机]负责读;
 &#39;REDIS_PORT&#39;=>&#39;6379&#39;,//端口号
 &#39;REDIS_TIMEOUT&#39;=>&#39;300&#39;,//超时时间
 &#39;REDIS_PERSISTENT&#39;=>false,//是否长连接 false=短连接
 &#39;REDIS_AUTH&#39;=>&#39;&#39;,//AUTH认证密码 
);
?>
로그인 후 복사

2.2. 실제 기능은 redis를 사용합니다:

/**
  * 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, 플래시 세일 핵심 문제는 대규모 동시성 경우 구매가 재고를 초과하지 않는다는 것입니다. 이것이 처리의 핵심이므로 기본 데이터 생성을 에서 수행하는 것이 좋습니다. 첫 번째 단계의 플래시 세일 카테고리:

//现在初始化里面定义后边要使用的redis参数
public function _initialize(){
  parent::_initialize();
  $goods_id = I("goods_id",&#39;0&#39;,&#39;intval&#39;);  
  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[&#39;uid&#39;];  
 }
로그인 후 복사

2.4, 두 번째 단계가 핵심입니다. 제품 세부 정보 페이지에 들어가기 전에 사용자는 먼저 현재 제품 재고를 대기열에 추가하고 다음과 같이 Redis에 저장합니다.

 /**
 * 访问产品前先将当前产品库存队列
 * @access public
 * @author bieanju
 */
 public function _before_detail(){
  $where[&#39;goods_id&#39;] = $this->goods_id;
  $where[&#39;start_time&#39;] = array("lt",time());
  $where[&#39;end_time&#39;] = array("gt",time());
  $goods = M("goods")->where($where)->field(&#39;goods_num,start_time,end_time&#39;)->find();
  !$goods && $this->error("当前秒杀已结束!");
  if($goods[&#39;goods_num&#39;] > $goods[&#39;order_num&#39;]){
   $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[&#39;goods_num&#39;]; $i ++) {
     $redis->lpush("{$this->goods_number_key}", 1);
    }
   }
   $resetRedis = $redis->llen("{$this->goods_number_key}");
   if(!$resetRedis){
    $this->error("系统繁忙,请稍后抢购!");
   }
  }else{
   $this->error("当前产品已经秒杀完!");
  }
   
 }
로그인 후 복사

다음으로 할 일은 ajax를 사용하여 사용자의 구매 버튼 클릭을 비동기식으로 처리하여 구매에 적합한 데이터를 대기열에 추가하는 것입니다(현재 사용자가 현재 제품 사용자의 대기열에 없으면 대기열에 들어가서 인벤토리 대기열을 팝업합니다. 입니다, 버리세요,):

/**
  * 抢购商品前处理当前会员是否进入队列
  * @access public
  * @author bieanju
  */
 public function goods_number_queue(){
  !$this->user_id && $this->ajaxReturn(array("status" => "-1","msg" => "请先登录"));
  $model = M("flash_sale");
  $where[&#39;goods_id&#39;] = $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[&#39;user_id&#39;] = $this->user_id;
    $condition[&#39;goods_id&#39;] = $this->goods_id;
    $condition[&#39;prom_type&#39;] = 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" => "系统繁忙,请重试!"));
  }
 }
로그인 후 복사

지정된 큐 값을 삭제하는 디버깅 기능 첨부:

 public function clearRedis(){
   set_time_limit(0);
   $redis = $this->connectRedis();
   //$Rd = $redis->del("{$this->user_queue_key}");
   $Rd = $redis->hDel("goods49",&#39;用户id&#39;&#39;);
   $a = $redis->hGet("goods_49_user", &#39;用户id&#39;);
   if(!$a){
    dump($a);
   }
   
   if($Rd == 0){
    exit("Redis队列已释放!");   
   }
  }
로그인 후 복사

위는 이 글의 내용입니다. 모든 내용은, 더 많은 관련 학습에 도움이 되길 바랍니다. 내용이 있으니 PHP 중국어 홈페이지를 주목해주세요!

관련 권장 사항:

ThinkPHP 및 Ajax는 보조 연결 드롭다운 메뉴를 구현합니다.


thinkphp의 URL 라우팅 규칙 및 정적 설정 정보

위 내용은 thinkphp와 redis, queue를 결합하여 구현한 코드의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿