swoole-做一頓飯來理解協程

協程

協程可以簡單理解爲線程,只不過這個線程是用戶態的,不需要操作系統參與,創建銷燬和切換的成本非常低,和線程不同的是協程沒法利用多核 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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章