協程
協程可以簡單理解爲線程,只不過這個線程是用戶態的,不需要操作系統參與,創建銷燬和切換的成本非常低,和線程不同的是協程沒法利用多核 cpu 的,想利用多核 cpu 需要依賴 Swoole 的多進程模型。—— swoole 協程一章
我的理解
可以把協程看成一道小學數學的一道題目:“合理安排時間”,來我們先做一道題目:
小明下班後回家煮飯,煲湯需要 10 分鐘,煮飯需要 8 分鐘, 炒菜需要 5 分鐘,,請問小明最少需要多少分鐘能煮好飯?
同步版煮飯
public function async()
{
$startTime = time();
echo "開始煲湯..." . PHP_EOL;
sleep(10);
echo "湯好了..." . PHP_EOL;
echo "開始煮飯..." . PHP_EOL;
sleep(8);
echo "飯熟了..." . PHP_EOL;
echo "放油..." . PHP_EOL;
sleep(1);
echo "煎魚..." . PHP_EOL;
sleep(3);
echo "放鹽..." . PHP_EOL;
sleep(1);
echo "出鍋..." . PHP_EOL;
var_dump('總耗時:' . (time() - $startTime) . ' 分鐘');
}
總耗時:23 分鐘
代碼很容易看懂,等待湯煮好之後再煮飯,然後再等待飯煮好再炒菜,生活中不會這樣操作吧?這就要引入協程來解決這個問題了。
協程版煮飯
<?php
namespace Study\Co;
use Swoole\Coroutine;
use Swoole\Coroutine\WaitGroup;
use Swoole;
class co
{
public function cookByCo()
{
$startTime = time();
// 開啓一鍵協程化: https://wiki.swoole.com/#/runtime?id=swoole_hook_all
Swoole\Runtime::enableCoroutine($flags = SWOOLE_HOOK_ALL);
// 創建一個協程容器: https://wiki.swoole.com/#/coroutine/scheduler
// 相當於進入廚房
\Co\run(function () {
// 等待結果: https://wiki.swoole.com/#/coroutine/wait_group?id=waitgroup
// 記錄哪道菜做好了,哪道菜還需要多長時間
$wg = new WaitGroup();
// 保存數據的結果
// 裝好的菜
$result = [];
// 記錄一下煲湯(記錄一個任務)
$wg->add();
// 創建一個煲湯任務(開啓一個新的協程)
Coroutine::create(function () use ($wg, &$result) {
echo "開始煲湯..." . PHP_EOL;
// 煲湯需要6分鐘,所以我們也不用在這裏等湯煮好,
// 直接去做下一個任務:炒菜(協程切換)
sleep(8);
echo "湯好了..." . PHP_EOL;
// 裝盤
$result['soup'] = '一鍋湯';
$wg->done(); // 標記任務完成
});
// 記錄一下煮飯(記錄一個任務)
$wg->add();
// 創建一個煮飯任務(開啓一個新的協程)
Coroutine::create(function () use ($wg, &$result) {
echo "開始煮飯..." . PHP_EOL;
// 煮飯需要5分鐘,所以我們不用在這裏等飯煮熟,放在這裏一會再來看看好了沒有
// 我們先去煲湯(協程切換)
sleep(10);
echo "飯熟了..." . PHP_EOL;
// 裝盤
$result['rice'] = '一鍋米飯';
$wg->done(); // 標記任務完成
});
// 記錄一下炒菜
$wg->add();
// 創建一個炒菜任務(再開啓一個新的協程)
Coroutine::create(function () use ($wg, &$result) {
// 煎魚的過程必須放在一個協程裏面執行,如果不是的話可能魚還沒煎好就出鍋了
// 因爲開啓協程後,IO全是異步了,在此demo中每次遇到sleep都會掛起當前協程
// 切換到下一個協程執行。
// 例如把出鍋這一步開啓一個新協程執行,則在煎魚的時候魚,魚就出鍋了。
echo "放油..." . PHP_EOL;
sleep(1);
echo "煎魚..." . PHP_EOL;
sleep(3);
echo "放鹽..." . PHP_EOL;
sleep(1);
echo "出鍋..." . PHP_EOL;
// 裝盤
$result['food'] = '魚香肉絲';
$wg->done();
});
// 等待全部任務完成
$wg->wait();
// 返回數據(上菜!)
var_dump($result);
});
var_dump('總耗時:' . (time() - $startTime) . ' 分鐘');
}
}
總耗時:10 分鐘
小明最少需要 10 分鐘能煮好飯。
主要參考過的文章
https://www.easyswoole.com/Cn/NoobCourse/coroutine.html
https://hyperf.wiki/#/zh-cn/coroutine
https://wiki.swoole.com/#/coroutine