swoole協程初探

yyy123456發表於2020-12-17

正在學習swoole的協程中,協程的特點是看起來象是併發執行。試驗一下。
本文前提,linux上已安裝了swoole。目前的最新版是 4.5.9(2020-12-17)

構造一個需求,一下建立5個協程,每個協程休眠1到3秒之間的隨機數,然後列印出睡眠時間,希望所有的程式結束時,整個耗時是最長3秒。

1.php

foreach (range(1, 5) as $v) {
    go(function () {
        $sleep_time = random_int(1, 3);
        sleep($sleep_time);
        echo "睡眠了" . $sleep_time . "秒\n";
    });
}

用php 1.php執行,程式能執行,但是感覺不像是協程,而是同步執行的。整個執行下來遠遠超過3秒。原來這裡有個小bug,應該使用 co:sleep, 這樣才會在本協程內睡眠,而不影響整個程式。


2.php

foreach (range(1, 5) as $v) {
    go(function () {
        $sleep_time = random_int(1, 3);
        co::sleep($sleep_time);
        echo "睡眠了" . $sleep_time . "秒\n";
    });
}

結果如下:

睡眠了1秒
睡眠了2秒
睡眠了2秒
睡眠了3秒
睡眠了3

總之一定是睡眠少的協程先退出,而且,睡眠了同樣時間的協程是同時列印出字元的。並且程式結束時的總共耗時就是3秒,說明併發成功。

現在希望5個協程結束後,能列印出hello world!


3.php

foreach (range(1, 5) as $v) {
    go(function () {
        $sleep_time = random_int(1, 3);
        co::sleep($sleep_time);
        $a = random_int(1, 1000);
        echo "睡眠了" . $sleep_time . "秒\n";
    });
}
echo "hello world!\n";

結果又出問題了,hello world 總是最先列印出來。
所以,這裡需要使用協程容器這個東東。


4.php

Co\run(function () {
    foreach (range(1, 5) as $v) {
        go(function ()  {
            $sleep_time = random_int(1, 3);
            co::sleep($sleep_time);
            echo "睡眠了" . $sleep_time . "秒\n";
        });
    }
});
echo "hello world!\n";

結果如下:

睡眠了1秒
睡眠了2秒
睡眠了3秒
睡眠了3秒
睡眠了3秒
hello world!

這次就能列印出正確結果了,perfect!
swoole的協程還是有點意思的~

本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章