校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > 關于Curl在Swoole協程中的解決方案詳析

關于Curl在Swoole協程中的解決方案詳析

熱門標簽:外呼系統撥打暫時無法接通 廣東地市地圖標注 高德地圖標注家 怎么向銷售公司推銷外呼系統 仁和怎么申請400開頭的電話 哪里辦理400電話 江西手機自動外呼防封系統是什么 廣州防封卡外呼系統多少錢一個月 長春人工外呼系統服務商

前言

眾所周知,在 Swoole 應用中,是不推薦使用 Curl 的,因為 Curl 會阻塞進程。

本文會用實際的代碼和數據,用最直觀的方式,讓你明白為什么。

最后還會給出 Curl 在 Swoole 中的解決方案,如果不想看分析可以直接拉到最后。

例程對比

宇潤看文章不喜歡那些虛的,所以自己寫也比較實在,直接來跑一下代碼,用數據看為什么不推薦在 Swoole 使用 Curl。

為了偷懶,我直接用了 YurunHttp 的 Curl 和 Swoole Handler,來替代那些又臭又長的 Curl 代碼。

代碼
composer.json

{
  "require": {
    "yurunsoft/yurun-http": "~3.0"
  }
}

server.php

?php
$http = new Swoole\Http\Server('127.0.0.1', 9501);
$http->on('workerstart', function(){
  \Swoole\Runtime::enableCoroutine();
});
$http->on('request', function ($request, $response) {
  sleep(1); // 假設各種處理耗時1秒
  $response->end($request->get['id'] . ': ' . date('Y-m-d H:i:s'));
});
$http->start();

test.php

?php

use Yurun\Util\YurunHttp;
use Yurun\Util\HttpRequest;

require __DIR__ . '/vendor/autoload.php';

define('REQUEST_COUNT', 3);

go(function(){
  // 協程客戶端
  echo 'coroutine http client:', PHP_EOL, PHP_EOL;
  $time = microtime(true);
  YurunHttp::setDefaultHandler(\Yurun\Util\YurunHttp\Handler\Swoole::class); // 切換為 Swoole Handler
  $channel = new \Swoole\Coroutine\Channel;
  for($i = 0; $i  REQUEST_COUNT; ++$i)
  {
    go(function() use($channel, $i){
      $http = new HttpRequest;
      $response = $http->get('http://127.0.0.1:9501/?id=' . $i); // 請求地址
      var_dump($response->body());
      $channel->push(1);
    });
  }
  for($i = 0; $i  REQUEST_COUNT; ++$i)
  {
    $channel->pop();
  }
  $channel->close();
  echo 'coroutine http client time: ', (microtime(true) - $time) . 's', PHP_EOL, PHP_EOL;

  // curl
  echo 'curl:', PHP_EOL, PHP_EOL;
  $time = microtime(true);
  YurunHttp::setDefaultHandler(\Yurun\Util\YurunHttp\Handler\Curl::class); // 切換為 Curl Handler
  $channel = new \Swoole\Coroutine\Channel;
  for($i = 0; $i  REQUEST_COUNT; ++$i)
  {
    go(function() use($channel, $i){
      $http = new HttpRequest;
      $response = $http->get('http://127.0.0.1:9501/?id=' . $i); // 請求地址
      var_dump($response->body());
      $channel->push(1);
    });
  }
  for($i = 0; $i  REQUEST_COUNT; ++$i)
  {
    $channel->pop();
  }
  $channel->close();
  echo 'curl time: ', (microtime(true) - $time) . 's', PHP_EOL, PHP_EOL;
});

運行

首次運行需要執行 composer update 安裝依賴

運行 php server.php,啟動服務端

運行 php test.php,啟動客戶端

運行結果

coroutine http client:

string(22) "1: 2019-09-11 08:35:54"
string(22) "0: 2019-09-11 08:35:54"
string(22) "2: 2019-09-11 08:35:54"
coroutine http client time: 1.0845630168915s

curl:

string(22) "0: 2019-09-11 08:35:55"
string(22) "1: 2019-09-11 08:35:56"
string(22) "2: 2019-09-11 08:35:57"
curl time: 3.0139901638031s

結果分析

上面的代碼在服務端延遲 1 秒后返回結果,模擬實際業務的耗時。

通過客戶端的耗時可以看出,Curl 3 次請求總共耗時 3 秒多,而協程客戶端僅耗時 1 秒多。

因為前一次請求中,Curl 等待返回內容的時間是干不了其他事情的。而協程客戶端等待返回內容期間,是掛起當前協程,轉而再去執行其它協程中的代碼。

解決方案

CoroutineHttpClient

使用 Swoole 內置的協程客戶端實現,適合有一定基礎的開發者使用。

文檔:https://wiki.swoole.com/wiki/...

Guzzle-Swoole

我們在項目中,可能很少直接寫 curl,但是用到的很多第三方類庫(如某某云們的 SDK)會有用到。

這些第三方類庫通常使用的是 Guzzle 作為 Http 客戶端,而 Guzzle 底層也是使用 Curl 實現。

宇潤專為此種場景研發了 Guzzle-Swoole 包,引入后可以讓這些 SDK 輕松支持協程,而不用修改一行代碼。

使用方法

執行命令直接安裝依賴:composer require yurunsoft/guzzle-swoole ~1.1

全局設定處理器:

?php
require dirname(__DIR__) . '/vendor/autoload.php';

use GuzzleHttp\Client;
use Yurun\Util\Swoole\Guzzle\SwooleHandler;
use GuzzleHttp\DefaultHandler;

DefaultHandler::setDefaultHandler(SwooleHandler::class);

go(function(){
  $client = new Client();
  $response = $client->request('GET', 'http://www.baidu.com', [
    'verify'  => false,
  ]);
  var_dump($response->getStatusCode());
});

手動指定 Swoole 處理器:

use GuzzleHttp\Client;
use GuzzleHttp\HandlerStack;
use Yurun\Util\Swoole\Guzzle\SwooleHandler;

go(function(){
  $handler = new SwooleHandler();
  $stack = HandlerStack::create($handler);
  $client = new Client(['handler' => $stack]);
  $response = $client->request('GET', 'http://www.baidu.com', [
    'verify'  => false,
  ]);
  var_dump($response->getBody()->__toString(), $response->getHeaders());
});

YurunHttp

YurunHttp 是開源的PHP HTTP類庫,支持鏈式操作,簡單易用。

支持所有常見的GET、POST、PUT、DELETE、UPDATE等請求方式,支持瀏覽器級別 Cookies 管理、上傳下載、設置和讀取header、Cookie、請求參數、失敗重試、限速、代理、證書等。

3.0 版完美支持Curl、Swoole 協程;3.2 版支持 Swoole WebSocket 客戶端。

使用方法

執行命令直接安裝依賴:composer require yurunsoft/yurun-http ~3.2

?php
use Yurun\Util\YurunHttp;
use Yurun\Util\HttpRequest;

// 設置默認請求處理器為 Swoole
YurunHttp::setDefaultHandler(\Yurun\Util\YurunHttp\Handler\Swoole::class);

// Swoole 處理器必須在協程中調用
go('test');

function test()
{
  $http = new HttpRequest;
  $response = $http->get('http://www.baidu.com');
  echo 'html:', PHP_EOL, $response->body();
}

截止發稿時,Swoole 4.4 新增的 hook Curl 依然是實驗性功能。雖然宇潤曾為該功能貢獻過一部分代碼,但是由于需要兼容的工作量非常大,有太多 OPTION 不被支持,我個人是暫時不推薦使用 hook Curl 的。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • 在Laravel5.6中使用Swoole的協程數據庫查詢
  • Swoole4.4協程搶占式調度器詳解

標簽:文山 黔東 梅河口 海北 廈門 惠州 湘西 濮陽

巨人網絡通訊聲明:本文標題《關于Curl在Swoole協程中的解決方案詳析》,本文關鍵詞  關于,Curl,在,Swoole,協程,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《關于Curl在Swoole協程中的解決方案詳析》相關的同類信息!
  • 本頁收集關于關于Curl在Swoole協程中的解決方案詳析的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 胶州市| 星子县| 海晏县| 永登县| 射阳县| 南漳县| 横山县| 浙江省| 长宁县| 淮南市| 蕲春县| 桐柏县| 渝北区| 德保县| 施秉县| 越西县| 尉氏县| 梓潼县| 来凤县| 阳高县| 榆林市| 左贡县| 宜宾市| 新昌县| 隆回县| 三台县| 连平县| 阿拉善左旗| 浙江省| 东宁县| 漳浦县| 旬邑县| 民勤县| 云林县| 溧水县| 班戈县| 武义县| 师宗县| 宜川县| 顺昌县| 彭山县|