ホームページ  >  記事  >  バックエンド開発  >  フラッシュ販売および注文獲得アクティビティを実装するための PHP マルチスレッド シミュレーション (コード付き)

フラッシュ販売および注文獲得アクティビティを実装するための PHP マルチスレッド シミュレーション (コード付き)

php中世界最好的语言
php中世界最好的语言オリジナル
2018-05-19 10:05:073225ブラウズ

今回は、フラッシュ販売と注文獲得活動を実現するための PHP マルチスレッド シミュレーション (コード付き) を紹介します。 フラッシュ販売と注文獲得活動を実現するための PHP マルチスレッド シミュレーションの 注意事項 は次のとおりです。ケース、見てみましょう。

最初にフラッシュ セール モジュールのアイデアについて説明します。

通常の状況でのユーザーのフラッシュ セール操作

1. フラッシュ セール リクエストを開始します
2. フラッシュ セールの結果を作成します。 1〜2秒のランダムなラグを伴うクエリリクエスト(偽装された迂回と見なすことができます)
4. 成功すると、注文が生成されます
5. 結果を返します

以下はフラッシュセールをシミュレートするコードです。 :

<?php
set_time_limit(0);
/**
* 线程的执行任务
*/
class Threadrun extends Thread
{
  public $url;
  public $data;
  public $params;
  public function construct($url, $params=[])
  {
   $this->url = $url;
   $this->params = $params;
  }
  public function run()
  {
   if(($url = $this->url))
   {
     $params = [
      'goods_id'  => 1,
      'activity_id'  => 1,
      'user_id'   => isset($this->params['user_id']) ? $this->params['user_id'] : $this->getCurrentThreadId(),
     ];
     $startTime = microtime(true);
     $this->data = [
      'id'   => $params['user_id'],
      'result'  => model_http_curl_get( $url, $params ),
      'time'  => microtime(true)-$startTime,
      'now'   => microtime(true),
     ];
   }
  }
}
/**
* 执行多线程
*/
function model_thread_result_get($urls_array)
{
  foreach ($urls_array as $key => $value)
  {
   $threadPool[$key] = new Threadrun($value["url"],['user_id'=>$value['user_id']]);
   $threadPool[$key]->start();
  }
  foreach ($threadPool as $thread_key => $thread_value)
  {
   while($threadPool[$thread_key]->isRunning())
   {
     usleep(10);
   }
   if($threadPool[$thread_key]->join())
   {
     $variable_data[$thread_key] = $threadPool[$thread_key]->data;
   }
  }
  return $variable_data;
}
/**
* 发送 HTTP 请求
*/
function model_http_curl_get($url,$data=[],$userAgent="")
{
  $userAgent = $userAgent ? $userAgent : 'Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.2)';
  $curl = curl_init();
  curl_setopt($curl, CURLOPT_URL, $url);
  curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
  curl_setopt($curl, CURLOPT_TIMEOUT, 5);
  curl_setopt($curl, CURLOPT_USERAGENT, $userAgent);
  curl_setopt($curl, CURLOPT_POST, true);
  if( !empty($data) ) {
   curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
  }
  $result = curl_exec($curl);
  curl_close($curl);
  return $result;
}
/**
 * 友好的打印变量
 * @param $val
 */
function dump( $val )
{
  echo '<pre class="brush:php;toolbar:false">';
  var_dump($val);
  echo '
'; } /**  * 写日志  * @param $msg  * @param string $logPath  */ function writeLog( $msg, $logPath='' ) {   if( empty($logPath) ) {    $logPath = date('Y_m_d').'.log';   }   if( !file_exists($logPath) ) {    $fp = fopen( $logPath,'w' );    fclose( $fp );   }   error_log( $msg.PHP_EOL, 3, $logPath); } /**  * 生成日志信息  * @param $result  * @param $timeDiff  * @return bool|string  */ function createLog( $result, $timeDiff ){   if( empty($result) || !is_array($result) ) {    return false;   }   $succeed = 0;   $fail = 0;   foreach( $result as $v ) {    $times[] = $v['time'];    $v['result'] === false ? $fail++ : $succeed++;   }   $totalTime = array_sum( $times );   $maxTime = max( $times );   $minTime = min( $times );   $sum = count( $times );   $avgTime = $totalTime/$sum;   $segment = str_repeat('=',100);   $flag = $segment . PHP_EOL;   $flag .= '总共执行时间:' . $timeDiff . PHP_EOL ;   $flag .= '最大执行时间:' . $maxTime . PHP_EOL;   $flag .= '最小执行时间:' . $minTime . PHP_EOL;   $flag .= '平均请求时间:' . $avgTime . PHP_EOL;   $flag .= '请求数:' . $sum . PHP_EOL;   $flag .= '请求成功数:' . $succeed . PHP_EOL;   $flag .= '请求失败数:' . $fail . PHP_EOL;   $flag .= $segment . PHP_EOL;   return $flag; } /**  * 发起秒杀请求  */ function insertList( $urls, $logPath='' ) {   $t = microtime(true);   $result = model_thread_result_get($urls);   $e = microtime(true);   $timeDiff = $e-$t;   echo "总执行时间:" . $timeDiff . PHP_EOL;   foreach( $result as $v ) {    $msg = '用户【' . $v['id'] . '】秒杀商品, 返回结果 ' . $v['result'] . ' 用时【' . $v['time'] . ' 秒】 当前时间【'.$v['now'].'】';    writeLog( $msg,$logPath );   }   $logStr = createLog( $result, $timeDiff);   writeLog( $logStr, $logPath );   return $result; } //发起秒杀请求 for ($i=0; $i < 1000; $i++) { $urls_array[] = array("name" => "baidu", "url" => "http://***.***.com/seckill/shopping/listinsert"); } $list = insertList( $urls_array, './inset.log' ); //发起秒杀结果查询请求 $urls_array = []; foreach( $list as $v ) {   if( $v['result'] === false ) {    continue;   }   $urls_array[] = array(         "name"  => "baidu",         "url"  => "http://***.***.com/seckill/shopping/query",         'user_id' => $v['id'],   ); } insertList( $urls_array, './query.log' );テスト コード マシンのパフォーマンス (開発マシン):

注文コード マシンのパフォーマンス (テスト マシン):

システム テスト結果:

1000 件の同時実行状況、1 秒あたり 300 以上の注文をシミュレート単一マシンなのでサーバーに負担がかかりません。

それどころか、テストマシンはそれに耐えられず、CPU が 100% 上昇しました。 Apache が時々クラッシュします。

PHPのマルチスレッドとWindows環境がそれをうまくサポートしていないためなのか、それともPHPのマルチスレッド自体に問題があるのか​​はわかりませんが、わずか1000スレッドでは実行できません。マルチスレッドには依然として Python と C が必要です。

この記事の事例を読んだ後は、この方法を習得したと思います。さらに興味深い情報については、php 中国語 Web サイトの他の関連記事に注目してください。

推奨読書:

Redis カウンタ クラスを使用する手順の詳細な説明

php のラッシュ購入クラスでの高同時実行リクエストの実装の詳細な説明

以上がフラッシュ販売および注文獲得アクティビティを実装するための PHP マルチスレッド シミュレーション (コード付き)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。