PHP沒有定時器?

確實,PHP沒有類似於JS中的setInterval或者setTimeout這樣的原生定時器相關的函數。但是我們可以通過其他方式來實現,比如使用declare。

先來看看是如何實現的,然後我們再好好學習一下declare表達式到底是個什麼東西。


function do_tick($str = '')
{
    list($sec, $usec) = explode(' ', microtime());
    printf("[%.4f] Tick.%s\n", $sec + $usec, $str);
}
register_tick_function('do_tick');

do_tick('--start--');
declare (ticks = 1) {
    while (1) {
        sleep(1); // 這裏,每執行一次就去調用一次do_tick()
    }
}

很簡單的代碼,運行起來以後將每秒輸出當前的時間。

declare語法的定義如下:

declare (directive)
    statemaent;
  • declare 結構用來設定一段代碼的執行指令
  • directive 部分允許設定 declare 代碼段的行爲。目前只認識兩個指令:ticks以及 encoding
  • Tick(時鐘週期)是一個在 declare 代碼段中解釋器每執行 N 條可計時的低級語句就會發生的事件。N 的值是在 declare 中的 directive 部分用 ticks=N 來指定的
  • 在每個 tick 中出現的事件是由 register_tick_function() 來指定的

這裏,我們只研究ticks的使用。

上述代碼中,我們使用register_tick_function()註冊了do_tick()方法給ticks,declare指定了ticks=1,也就是每執行一次可計時的低級語句,就會去執行一次register_tick_function()中註冊的方法。當declare代碼塊中的while每次循環時,都有一個sleep()停頓了一秒,而這個sleep()就是那個可計時的低級語句。

那麼,while()不是可計時的低級語句嘛?當然不是,where、if等條件判斷都不是這種可計時的低級語句。

不是所有語句都可計時。通常條件表達式和參數表達式都不可計時。

我們通過下面這個例子再來看看具體到一步步declare是怎樣執行的:


function test_tick()
{
    static $i = 0;
    echo 'test_tick:' . $i++, PHP_EOL;
}
register_tick_function('test_tick');
test_tick(); // test_tick:0

$j = 0; 
declare (ticks = 1) {
    $j++; // test_tick:1

    $j++; // test_tick: 2
    
    sleep(1); //  停1秒後,test_tick:3

    $j++; // test_tick:4

    if ($j == 3) { // 條件表達式,不會執行ticks

        echo "aa", PHP_EOL; // test_tick:5 \n   test_tick:6,PHP_EOL會計一次ticks
    }
}

// declare使用花括號後面所有代碼無效果,作用域限定在花括號以內
echo "bbb"; // 
echo "ccc"; // 
echo "ddd"; // 

註釋很詳細了,我們就不用一一說明了。下面我們來看將ticks定爲2,並且declare下面的statemaent不用花括號的結果:


function test_tick1() 
{
    static $i = 0;
    echo 'test_tick1:' . $i++, PHP_EOL;
}
register_tick_function('test_tick1');

$j = 0; // 此處不計時
declare (ticks = 2); 
$j++; // test_tick1:0 

$j++; 

sleep(1); //  停1秒後 test_tick1:1

$j++; 

$j++; // test_tick1:2

if ($j == 4) { // 條件表達式,不會執行ticks
    // echo "aa", PHP_EOL;
    echo "aa"; // test_tick:10,test_tick1不執行,沒有跳兩步,如果用了,PHP_EOL,那麼算兩步,會輸出test_tick1:3
}

//  declare沒有使用花括號將對後面所有代碼起效果,如果是require或者include將不會對父頁面後續內容進行處理
echo "bbb"; // test_tick1:3
echo "ccc";
echo "ddd"; // test_tick1:4

可以看出,我們declare對其定義後續的代碼都產生了作用,但需要注意的是如果有頁面嵌套,對父頁面的後續代碼是沒有效果的。而定義了ticks=2之後,將在兩個低級可計時代碼後執行一次register_tick_function()註冊的函數代碼。

測試代碼:https://github.com/zhangyue0503/dev-blog/blob/master/php/201911/source/PHP%E6%B2%A1%E6%9C%89%E5%AE%9A%E6%97%B6%E5%99%A8%EF%BC%9F.php

參考文檔:https://www.php.net/manual/zh/control-structures.declare.php

===========

各自媒體平臺均可搜索【硬核項目經理】

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