亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > PHP > 正文

php結合redis實現高并發下的搶購、秒殺功能

2019-11-08 03:00:29
字體:
來源:轉載
供稿:網友

本文轉自http://blog.csdn.net/nuli888/article/details/51865401

搶購、秒殺是如今很常見的一個應用場景,主要需要解決的問題有兩個:

1 高并發對數據庫產生的壓力2 競爭狀態下如何解決庫存的正確減少("超賣"問題)對于第一個問題,已經很容易想到用緩存來處理搶購,避免直接操作數據庫,例如使用Redis。重點在于第二個問題

常規寫法:

查詢出對應商品的庫存,看是否大于0,然后執行生成訂單等操作,但是在判斷庫存是否大于0處,如果在高并發下就會有問題,導致庫存量出現負數

[php] view plain copy <?php  $conn=MySQL_connect("localhost","big","123456");    if(!$conn){        echo "connect failed";        exit;    }   mysql_select_db("big",$conn);   mysql_query("set names utf8");    $優化方案1:將庫存字段number字段設為unsigned,當庫存為0時,因為字段不能為負數,將會返回false

[php] view plain copy //庫存減少  $sql="update ih_store set number=number-{$number} where sku_id='$sku_id' and number>0";  $store_rs=mysql_query($sql,$conn);    if(mysql_affected_rows()){        insertLog('庫存減少成功');  }  

優化方案2:使用MySQL的事務,鎖住操作的行

[php] view plain copy <?php  $conn=mysql_connect("localhost","big","123456");    if(!$conn){        echo "connect failed";        exit;    }   mysql_select_db("big",$conn);   mysql_query("set names utf8");    $price=10;  $user_id=1;  $goods_id=1;  $sku_id=11;  $number=1;    //生成唯一訂單號  function build_order_no(){      return date('ymd').substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 8);  }  //記錄日志  function insertLog($event,$type=0){      global $conn;      $sql="insert into ih_log(event,type)       values('$event','$type')";        mysql_query($sql,$conn);    }    //模擬下單操作  //庫存是否大于0  mysql_query("BEGIN");   //開始事務  $sql="select number from ih_store where goods_id='$goods_id' and sku_id='$sku_id' FOR UPDATE";//此時這條記錄被鎖住,其它事務必須等待此次事務提交后才能執行  $rs=mysql_query($sql,$conn);  $row=mysql_fetch_assoc($rs);  if($row['number']>0){      //生成訂單       $order_sn=build_order_no();       $sql="insert into ih_order(order_sn,user_id,goods_id,sku_id,price)       values('$order_sn','$user_id','$goods_id','$sku_id','$price')";        $order_rs=mysql_query($sql,$conn);             //庫存減少      $sql="update ih_store set number=number-{$number} where sku_id='$sku_id'";      $store_rs=mysql_query($sql,$conn);        if(mysql_affected_rows()){            insertLog('庫存減少成功');          mysql_query("COMMIT");//事務提交即解鎖      }else{            insertLog('庫存減少失敗');      }  }else{      insertLog('庫存不夠');      mysql_query("ROLLBACK");  }  ?>  

優化方案3:使用非阻塞的文件排他鎖

[php] view plain copy <?php  $conn=mysql_connect("localhost","root","123456");    if(!$conn){        echo "connect failed";        exit;    }   mysql_select_db("big-bak",$conn);   mysql_query("set names utf8");    $price=10;  $user_id=1;  $goods_id=1;  $sku_id=11;  $number=1;    //生成唯一訂單號  function build_order_no(){      return date('ymd').substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 8);  }  //記錄日志  function insertLog($event,$type=0){      global $conn;      $sql="insert into ih_log(event,type)       values('$event','$type')";        mysql_query($sql,$conn);    }    $fp = fopen("lock.txt", "w+");  if(!flock($fp,LOCK_EX | LOCK_NB)){      echo "系統繁忙,請稍后再試";      return;  }  //下單  $sql="select number from ih_store where goods_id='$goods_id' and sku_id='$sku_id'";  $rs=mysql_query($sql,$conn);  $row=mysql_fetch_assoc($rs);  if($row['number']>0){//庫存是否大于0      //模擬下單操作       $order_sn=build_order_no();       $sql="insert into ih_order(order_sn,user_id,goods_id,sku_id,price)       values('$order_sn','$user_id','$goods_id','$sku_id','$price')";        $order_rs=mysql_query($sql,$conn);             //庫存減少      $sql="update ih_store set number=number-{$number} where sku_id='$sku_id'";      $store_rs=mysql_query($sql,$conn);        if(mysql_affected_rows()){            insertLog('庫存減少成功');          flock($fp,LOCK_UN);//釋放鎖      }else{            insertLog('庫存減少失敗');      }   }else{      insertLog('庫存不夠');  }  fclose($fp);  優化方案4:使用redis隊列,因為pop操作是原子的,即使有很多用戶同時到達,也是依次執行,推薦使用(mysql事務在高并發下性能下降很厲害,文件鎖的方式也是)

先將商品庫存如隊列

[php] view plain copy <?php  $store=1000;  $redis=new Redis();  $result=$redis->connect('127.0.0.1',6379);  $res=$redis->llen('goods_store');  echo $res;  $count=$store-$res;  for($i=0;$i<$count;$i++){      $redis->lpush('goods_store',1);  }  echo $redis->llen('goods_store');  ?>  搶購、描述邏輯[php] view plain copy <?php  $conn=mysql_connect("localhost","big","123456");    if(!$conn){        echo "connect failed";        exit;    }   mysql_select_db("big",$conn);   mysql_query("set names utf8");    $price=10;  $user_id=1;  $goods_id=1;  $sku_id=11;  $number=1;    //生成唯一訂單號  function build_order_no(){      return date('ymd').substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 8);  }  //記錄日志  function insertLog($event,$type=0){      global $conn;      $sql="insert into ih_log(event,type)       values('$event','$type')";        mysql_query($sql,$conn);    }    //模擬下單操作  //下單前判斷redis隊列庫存量  $redis=new Redis();  $result=$redis->connect('127.0.0.1',6379);  $count=$redis->lpop('goods_store');  if(!$count){      insertLog('error:no store redis');      return;  }    //生成訂單    $order_sn=build_order_no();  $sql="insert into ih_order(order_sn,user_id,goods_id,sku_id,price)   values('$order_sn','$user_id','$goods_id','$sku_id','$price')";    $order_rs=mysql_query($sql,$conn);     //庫存減少  $sql="update ih_store set number=number-{$number} where sku_id='$sku_id'";  $store_rs=mysql_query($sql,$conn);    if(mysql_affected_rows()){        insertLog('庫存減少成功');  }else{        insertLog('庫存減少失敗');  }   模擬5000高并發測試webbench -c 5000 -t 60 http://192.168.1.198/big/index.phpab -r -n 6000 -c 5000  http://192.168.1.198/big/index.php

上述只是簡單模擬高并發下的搶購,真實場景要比這復雜很多,很多注意的地方如搶購頁面做成靜態的,通過Ajax調用接口再如上面的會導致一個用戶搶多個,思路:需要一個排隊隊列和搶購結果隊列及庫存隊列。高并發情況,先將用戶進入排隊隊列,用一個線程循環處理從排隊隊列取出一個用戶,判斷用戶是否已在搶購結果隊列,如果在,則已搶購,否則未搶購,庫存減1,寫數據庫,將用戶入結果隊列。

測試數據表

[php] view plain copy --  -- 數據庫: `big`  --    -- --------------------------------------------------------    --  -- 表的結構 `ih_goods`  --      CREATE TABLE IF NOT EXISTS `ih_goods` (    `goods_id` int(10) unsigned NOT NULL AUTO_INCREMENT,    `cat_id` int(11) NOT NULL,    `goods_name` varchar(255) NOT NULL,    PRIMARY KEY (`goods_id`)  ) ENGINE=MyISAM  DEFAULT CHARSET=utf8 AUTO_INCREMENT=2 ;      --  -- 轉存表中的數據 `ih_goods`  --      INSERT INTO `ih_goods` (`goods_id`, `cat_id`, `goods_name`) VALUES  (1, 0, '小米手機');    -- --------------------------------------------------------    --  -- 表的結構 `ih_log`  --    CREATE TABLE IF NOT EXISTS `ih_log` (    `id` int(11) NOT NULL AUTO_INCREMENT,    `event` varchar(255) NOT NULL,    `type` tinyint(4) NOT NULL DEFAULT '0',    `addtime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,    PRIMARY KEY (`id`)  ) ENGINE=MyISAM DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;    --  -- 轉存表中的數據 `ih_log`  --      -- --------------------------------------------------------    --  -- 表的結構 `ih_order`  --    CREATE TABLE IF NOT EXISTS `ih_order` (    `id` int(11) NOT NULL AUTO_INCREMENT,    `order_sn` char(32) NOT NULL,    `user_id` int(11) NOT NULL,    `status` int(11) NOT NULL DEFAULT '0',    `goods_id` int(11) NOT NULL DEFAULT '0',    `sku_id` int(11) NOT NULL DEFAULT '0',    `price` float NOT NULL,    `addtime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,    PRIMARY KEY (`id`)  ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='訂單表' AUTO_INCREMENT=1 ;    --  -- 轉存表中的數據 `ih_order`  --      -- --------------------------------------------------------    --  -- 表的結構 `ih_store`  --    CREATE TABLE IF NOT EXISTS `ih_store` (    `id` int(11) NOT NULL AUTO_INCREMENT,    `goods_id` int(11) NOT NULL,    `sku_id` int(10) unsigned NOT NULL DEFAULT '0',    `number` int(10) NOT NULL DEFAULT '0',    `freez` int(11) NOT NULL DEFAULT '0' COMMENT '虛擬庫存',    PRIMARY KEY (`id`)  ) ENGINE=InnoDB  DEFAULT CHARSET=utf8 COMMENT='庫存' AUTO_INCREMENT=2 ;    --  -- 轉存表中的數據 `ih_store`  --    INSERT INTO `ih_store` (`id`, `goods_id`, `sku_id`, `number`, `freez`) VALUES  (1, 1, 11, 500, 0); 
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲精品一区在线观看香蕉| 亚洲国产精品福利| 日日噜噜噜夜夜爽亚洲精品| 91色视频在线观看| 欧美亚洲伦理www| 91国产精品电影| 日韩免费av一区二区| 久久久亚洲国产天美传媒修理工| 日本在线精品视频| 全色精品综合影院| 久久久久五月天| 日韩精品亚洲元码| 久久久国产在线视频| 日本高清不卡在线| 成人性教育视频在线观看| 国产在线视频2019最新视频| 4388成人网| 韩国福利视频一区| 亚洲综合色激情五月| 亚洲成年人影院在线| 国产精品综合网站| 国产成人91久久精品| 亚洲精品有码在线| 日韩经典第一页| 国产欧美日韩高清| 国产精品一二区| 久久亚洲一区二区三区四区五区高| 亚洲va男人天堂| 日韩精品极品在线观看播放免费视频| 在线视频日本亚洲性| 亚洲欧美国产日韩天堂区| 日韩免费av片在线观看| 久久久www成人免费精品| 色综合久久88色综合天天看泰| 国产在线精品播放| 亚洲伊人第一页| www.亚洲人.com| 欧美又大又粗又长| 欧美色xxxx| 亚洲欧美一区二区精品久久久| 国产成人精品免高潮费视频| 九九热在线精品视频| 欧美黄色三级网站| 欧美日韩国产区| 一区二区三区久久精品| 91免费看国产| 亚洲专区中文字幕| 黄色精品一区二区| 亚洲精品一区二区三区婷婷月| 亚洲精品电影网| 成人午夜黄色影院| 欧美综合一区第一页| 色噜噜久久综合伊人一本| 欧美精品18videosex性欧美| 欧美日韩国产中字| 色噜噜国产精品视频一区二区| 国产婷婷97碰碰久久人人蜜臀| 亚洲开心激情网| 欧美中文字幕在线观看| 国产香蕉97碰碰久久人人| 欧美激情aaaa| 久久国产精品久久久久| 色综合久久中文字幕综合网小说| 精品久久中文字幕| 日韩美女视频中文字幕| 国产亚洲激情视频在线| 国产经典一区二区| 97国产精品久久| 国产精品美女呻吟| 久久久久中文字幕| 国产91精品久久久久久| 国产精品久久久久久久av大片| 伊人久久久久久久久久久久久| 日韩欧美成人网| 亚洲性视频网站| 成人免费在线网址| 色哟哟亚洲精品一区二区| 久久资源免费视频| 国产成人综合av| 色综合久久久久久中文网| 亚洲欧美另类国产| 亚洲第一级黄色片| 精品丝袜一区二区三区| 国产精品久久二区| 国产精品高潮呻吟视频| 亚洲成人xxx| 午夜精品久久久久久久99黑人| 国产精品丝袜白浆摸在线| 欧美日韩另类在线| 国产精品69久久| 国产人妖伪娘一区91| 国产成人自拍视频在线观看| 欧美日韩亚洲91| 亚洲美女黄色片| 中文字幕国产亚洲| 亚洲精品一区在线观看香蕉| 欧美性在线观看| 国产精品久久久久免费a∨大胸| 亚洲网址你懂得| 国产精品自产拍高潮在线观看| 91精品国产成人| 成人疯狂猛交xxx| 欧洲s码亚洲m码精品一区| 色综合天天狠天天透天天伊人| 欧日韩在线观看| 精品激情国产视频| 日韩av在线影视| 欧美日韩国产一中文字不卡| 国产免费一区视频观看免费| 亚洲欧美日韩中文在线| 日韩在线视频导航| 麻豆乱码国产一区二区三区| 成年人精品视频| 日本一区二区在线免费播放| 精品国产视频在线| 宅男66日本亚洲欧美视频| 日本中文字幕不卡免费| 成人伊人精品色xxxx视频| 亚洲性69xxxbbb| 国产精品久久久91| 久99久在线视频| 91精品国产91久久久久久久久| 91高清在线免费观看| 欧美在线视频一区| 精品视频在线播放色网色视频| 福利精品视频在线| 色yeye香蕉凹凸一区二区av| 亚洲在线视频福利| 国产视频自拍一区| 欧美精品少妇videofree| 一本色道久久88综合亚洲精品ⅰ| 欧美日产国产成人免费图片| 亚洲国产精品va在线看黑人| 日韩av手机在线| 亚洲美女黄色片| 欧美疯狂做受xxxx高潮| 亚洲综合第一页| 午夜精品久久久久久久99黑人| 国产成人精品在线| 久久天天躁狠狠躁夜夜av| 国产999视频| 福利一区福利二区微拍刺激| 久久97精品久久久久久久不卡| 国产精品视频免费在线观看| 午夜精品美女自拍福到在线| 色多多国产成人永久免费网站| 日韩欧美亚洲综合| 亚洲欧美另类在线观看| 欧美黄色三级网站| 成人午夜激情网| 欧美午夜丰满在线18影院| 国产精品免费视频久久久| 国产精品福利久久久| 日韩的一区二区| 久久在线免费视频| 欧美中文在线免费| 欧美精品在线视频观看| 日本午夜人人精品| 日韩欧美黄色动漫| 亚洲色图激情小说| 日韩欧美亚洲一二三区| 久久999免费视频| 91精品国产综合久久香蕉最新版| 日本19禁啪啪免费观看www|