濮阳杆衣贸易有限公司

主頁(yè) > 知識(shí)庫(kù) > php處理?yè)屬?gòu)類功能的高并發(fā)請(qǐng)求

php處理?yè)屬?gòu)類功能的高并發(fā)請(qǐng)求

熱門標(biāo)簽:南寧高頻外呼回?fù)芟到y(tǒng)哪家好 400電話辦理福州市 長(zhǎng)沙crm外呼系統(tǒng)業(yè)務(wù) 深圳外呼系統(tǒng)收費(fèi) 江蘇外呼電銷機(jī)器人報(bào)價(jià) 400電話申請(qǐng)方法收費(fèi) 離石地圖標(biāo)注 專業(yè)電話機(jī)器人批發(fā)商 電話機(jī)器人危險(xiǎn)嗎

本文以搶購(gòu)、秒殺為例。介紹如何在高并發(fā)狀況下確保數(shù)據(jù)正確。
在高并發(fā)請(qǐng)求下容易參數(shù)兩個(gè)問(wèn)題
1.數(shù)據(jù)出錯(cuò),導(dǎo)致產(chǎn)品超賣。
2.頻繁操作數(shù)據(jù)庫(kù),導(dǎo)致性能下降。

測(cè)試環(huán)境

Windows7
apache2.4.9
php5.5.12
php框架 yii2.0
工具 apache bench (apache自帶高并發(fā)請(qǐng)求工具)。

通常處理方法

從控制器可以看出代碼思路。先查詢商品庫(kù)存。如果庫(kù)存大于0 ,則庫(kù)存減少1,同時(shí)生產(chǎn)訂單,錄入搶購(gòu)者數(shù)據(jù)。

// 常規(guī)代碼處理高并發(fā)
  public function actionNormal(){
    // 查詢庫(kù)存
    $stock = Goods::find()->select('stock')->where(['goods_id'=>100001])->asArray()->one();
    // 判斷該商品是否還有庫(kù)存
    if ($stock['stock']>0) {
      // 庫(kù)存減一
      Goods::updateAllCounters(['stock' => -1],['goods_id'=>100001]);

      // 生產(chǎn)訂單(另外功能,暫且隨機(jī)賦值)
      $order = $this->build_order();

      // 秒殺信息入庫(kù)
      $model = new Highly();
      $model->order_id = $order;
      $model->goods_name = '秒殺商品';
      $model->buy_time = date('Y-m-d H:i:s',time());
      $model->mircrotime = microtime(true);
      if($model->save()===false){
        echo '未能成功搶購(gòu)!';
      }else{
        echo '恭喜你,訂單b>'.$order.'/b>搶購(gòu)成功';
      }

    }else{
      echo '已被搶購(gòu)一空!';
    }
  }

將商品庫(kù)存設(shè)置為20后,通過(guò)ab 配置200的并發(fā)請(qǐng)求。

ab -n 200 -c 200 http//localhost/highly/normal

執(zhí)行結(jié)果發(fā)現(xiàn)庫(kù)存變成了負(fù)值,商品超賣了。

原因比較簡(jiǎn)單,在高并發(fā)請(qǐng)求下。在生產(chǎn)訂單,減少庫(kù)存之前,會(huì)優(yōu)先查詢到庫(kù)存結(jié)果。

優(yōu)化一:修改庫(kù)存數(shù)據(jù)類型

第一種優(yōu)化方法,從數(shù)據(jù)庫(kù)入手。既然查詢到的結(jié)果不準(zhǔn)確,那我就在庫(kù)存減少上做手腳。將庫(kù)存的數(shù)據(jù)類型改成無(wú)符號(hào)(不能有負(fù)值)。

代碼還是跟上面差不多,只是在庫(kù)存減1的地方做了個(gè)判斷。避免報(bào)錯(cuò)。

public function actionNormal(){
    // 查詢庫(kù)存
    $stock = Goods::find()->select('stock')->where(['goods_id'=>100001])->asArray()->one();
    // 判斷該商品是否還有庫(kù)存
    if ($stock['stock']>0) {
      // 庫(kù)存減一
      if(Goods::updateAllCounters(['stock' => -1],['goods_id'=>100001])===false){
        echo "已被搶購(gòu)一空!";
        return false;
      }

      // 生產(chǎn)訂單(另外功能,暫且隨機(jī)賦值)
      $order = $this->build_order();

      // 秒殺信息入庫(kù)
      $model = new Highly();
      $model->order_id = $order;
      $model->goods_name = '秒殺商品';
      $model->buy_time = date('Y-m-d H:i:s',time());
      $model->mircrotime = microtime(true);
      if($model->save()===false){
        echo '未能成功搶購(gòu)!';
      }else{
        echo '恭喜你,訂單b>'.$order.'/b>搶購(gòu)成功';
      }

    }else{
      echo '已被搶購(gòu)一空!';
    }
  }

這一次同樣200的并發(fā),執(zhí)行結(jié)果發(fā)現(xiàn)。數(shù)據(jù)正確,并不會(huì)出現(xiàn)超賣的情況。
思路其實(shí)也比較簡(jiǎn)單。因?yàn)閹?kù)存不能為負(fù)值,當(dāng)庫(kù)存等于0時(shí),如果還有值傳進(jìn)來(lái),則會(huì)報(bào)錯(cuò)。請(qǐng)求被終止。

這種優(yōu)化方式,雖然避免了商品超賣的情況。但是在另一方面,請(qǐng)求仍然會(huì)對(duì)數(shù)據(jù)庫(kù)造成壓力。如果多個(gè)功能使用此數(shù)據(jù)庫(kù),會(huì)造成性能下降厲害。

優(yōu)化二:redis

利用 redis list類型的pop的原子性。在操作數(shù)據(jù)庫(kù)前,做一個(gè)驗(yàn)證。當(dāng)商品賣完后,就不允許再繼續(xù)進(jìn)行數(shù)據(jù)庫(kù)操作。

// redis list 高并發(fā)測(cè)試
  public function actionRedis(){
    $redis = \Yii::$app->redis;
    // $redis->lpush('mytest',1);
    $order = $this->build_order();
    // echo $order;die;
    // echo $redis->llen('mytest');
    $reg = $redis->lpop('mytest');
    if (!$reg) {
      echo "笨蛋!已經(jīng)被搶光啦!";
      return false;
    }
    $redis->close();
    $model = new Highly();
    $model->order_id = $order;
    $model->goods_name = '秒殺商品';
    $model->buy_time = date('Y-m-d H:i:s',time());
    $model->mircrotime = microtime(true);

    if($model->save()===false){
      echo '未能成功搶購(gòu)!';
    }else{
      echo '恭喜你,訂單b>'.$order.'/b>搶購(gòu)成功';
    }
  }
  // 給redis添加商品
  public function actionInsertgoods(){
    $count = yii::$app->request->get('count',0);
    if (empty($count)) {
      echo '大兄弟,你還沒(méi)告訴我需要上架多少商品呢!';
      return false;
    }
    $redis = \Yii::$app->redis;
    for ($i=0; $i  $count; $i++) { 
      $redis->lpush('mytest',1);
    }
    echo '成功添加了'.$redis->llen('mytest').'件商品。';
    $redis->close();

  }

這點(diǎn)的代碼,我寫了兩個(gè)方法。第一個(gè)方法是秒殺的代碼,第二個(gè)方法是給秒殺的商品設(shè)置數(shù)量。為了方便測(cè)試,我這里處理的比較簡(jiǎn)單。

通過(guò)測(cè)試,數(shù)據(jù)庫(kù)生產(chǎn)的訂單數(shù)量正常,并沒(méi)有出現(xiàn)問(wèn)題。而又避免了請(qǐng)求數(shù)據(jù)庫(kù)造成性能下降的問(wèn)題。同時(shí)內(nèi)存數(shù)據(jù)庫(kù)redis查詢的速度要比mysql快很多。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • PHP并發(fā)場(chǎng)景的三種解決方案代碼實(shí)例
  • php并發(fā)加鎖問(wèn)題分析與設(shè)計(jì)代碼實(shí)例講解
  • PHP解決高并發(fā)的優(yōu)化方案實(shí)例
  • php多進(jìn)程模擬并發(fā)事務(wù)產(chǎn)生的問(wèn)題小結(jié)
  • PHP利用Mysql鎖解決高并發(fā)的方法
  • php curl批處理實(shí)現(xiàn)可控并發(fā)異步操作示例
  • PHP+Redis 消息隊(duì)列 實(shí)現(xiàn)高并發(fā)下注冊(cè)人數(shù)統(tǒng)計(jì)的實(shí)例
  • PHP開(kāi)發(fā)中解決并發(fā)問(wèn)題的幾種實(shí)現(xiàn)方法分析
  • PHP使用Redis實(shí)現(xiàn)防止大并發(fā)下二次寫入的方法
  • php結(jié)合redis高并發(fā)下發(fā)帖、發(fā)微博的實(shí)現(xiàn)方法
  • 詳解php處理大并發(fā)大流量大存儲(chǔ)

標(biāo)簽:南昌 濱州 興安盟 白酒營(yíng)銷 南京 曲靖 太原 株洲

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《php處理?yè)屬?gòu)類功能的高并發(fā)請(qǐng)求》,本文關(guān)鍵詞  php,處理,搶購(gòu),類,功能,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《php處理?yè)屬?gòu)類功能的高并發(fā)請(qǐng)求》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于php處理?yè)屬?gòu)類功能的高并發(fā)請(qǐng)求的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    庆元县| 滕州市| 卢氏县| 孟连| 拜泉县| 会宁县| 环江| 香河县| 阳泉市| 开江县| 宝清县| 柞水县| 闵行区| 石河子市| 宜君县| 射阳县| 资兴市| 和田市| 沭阳县| 邵东县| 永寿县| 象山县| 揭东县| 宣威市| 天全县| 阳江市| 正安县| 枣强县| 通辽市| 连江县| 阿尔山市| 江都市| 满城县| 施甸县| 株洲县| 和顺县| 米泉市| 五大连池市| 锡林浩特市| 剑河县| 怀宁县|