這篇文章給大家分享的是有關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.0845630168915scurl:
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 的。
感謝各位的閱讀!關于“Curl在Swoole協程中有什么用”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。