PHP多進程學習(三)__代碼案例來了解父進程與子進程的執行順序


pcntl_fork創建子進程成功的話,系統就有了2個進程,一個為父進程,一個為子進程,父進程和子進程都繼續向下執行,子進程的id號為$pid(父進程會獲取子進程的$pid也就是$pid不為0,而子進程會獲取$pid為0)在系統運行到$pid = pcntl_fork();時,在這個地方進行分支,父子進程各自開始運行各自的程序代碼(通過if else語句判斷$pid我們就可以在指定位置寫上不同的邏輯代碼)。

代碼的運行結果是parent 和child,很奇怪吧,為什么一個if和else互斥的代碼中,都輸出了結果?其實是像上邊所說的,代碼在pcntl_fork時,一個父進程運行parent,一個子進程運行了child。在代碼結果上就顯示了parent和child。至於誰先誰后的問題,這得要看系統資源的分配了。

案例①

<?php $pid = pcntl_fork(); if($pid == -1) { //錯誤處理:創建子進程失敗時返回-1.
    die('fork error'); } else if ($pid) { //父進程會得到子進程號,所以這里是父進程執行的邏輯
    echo "parent \n"; //等待子進程中斷,防止子進程成為僵屍進程。
 pcntl_wait($status); } else { //子進程得到的$pid為0, 所以這里是子進程執行的邏輯。
    echo "child \n"; exit; }
# php fork.php       //結果
parent 
child 

上述代碼會經過多次執行,分別輸出parent和child。貌似是優先執行父進程。那么輸出的parent和child是否會有順序之分?是父進程會先執行?下面繼續測試。。。

案例②

<?php $pid = pcntl_fork(); if($pid == -1) { die('fork error'); } else if ($pid) { sleep(3); echo "parent \n"; pcntl_wait($status); } else { echo "child \n"; exit; }
# php fork.php  //結果
child 
parent 

我們在父進程中通過sleep來延緩執行,看看效果。
結果是,很快輸出了child,等待了接近3秒后,才輸出parent。所以父進程和子進程的執行是相對獨立的,沒有先后之分。

那么問題又來了?pcntl_wait是做什么用的?
會掛起當前進程,直到子進程退出,如果子進程在調用此函數之前就已退出,此函數會立刻返回。子進程使用的資源將被釋放。

案例③

<?php $pid = pcntl_fork(); if($pid == -1) { die('fork error'); } else if ($pid) { pcntl_wait ($status); echo "parent \n"; } else { sleep(3); echo "child \n"; exit; }
# php fork.php  //結果 
child parent 

上述代碼,我們可以看到,父進程執行pcntl_wait時就已經掛起,直到等待3秒后輸出child,子進程退出后。父進程繼續執行,輸出parent。

案例④

<?php define('FORK_NUMS', 3); $pids = array(); for($i = 0; $i < FORK_NUMS; ++$i) { $pids[$i] = pcntl_fork(); if($pids[$i] == -1) { die('fork error'); } else if ($pids[$i]) { pcntl_waitpid($pids[$i], $status); echo "pernet \n"; } else { sleep(3); echo "child id:" . getmypid() . " \n"; exit; } }
# php fork.php  //結果 
child id:57534 pernet child id:57538 pernet child id:57540 pernet 

上述代碼,我們創建3個子進程,父進程分別掛起等待子進程結束后,輸出parent。

案例⑤

<?php define('FORK_NUMS', 3); $pids = array(); for($i = 0; $i < FORK_NUMS; ++$i) { $pids[$i] = pcntl_fork(); if($pids[$i] == -1) { die('fork error'); } else if ($pids[$i]) { } else { sleep(3); echo "child id:" . getmypid() . " \n"; exit; } } foreach($pids as $k => $v) { if($v) { pcntl_waitpid($v, $status); echo "parent \n"; } }
# php fork.php  //結果
child id:57579 child id:57581 child id:57580 parent parent parent 

為什么上述代碼跟例4的輸出結果不一樣?
我們可以看到案例⑤的pcntl_waitpid函數放在了foreach中,foreach代碼是在主進程中,也就是父進程的代碼中。當執行foreach時,可能子進程已經全部執行完畢並退出。pcntl_waitpid會立刻返回,連續輸出三個parent。
在子進程中,需通過exit來退出,不然會產生遞歸多進程,父進程中不需要exit,不然會中斷多進程。

案例⑥

<?php define('FORK_NUMS', 3); $pids = array(); $fp = fopen('./test.log', 'wb'); $num = 1; for($i = 0; $i < FORK_NUMS; ++$i) { $pids[$i] = pcntl_fork(); if($pids[$i] == -1) { die('fork error'); } else if ($pids[$i]) { } else { for($i = 0; $i < 5; ++$i) { flock($fp, LOCK_EX); fwrite($fp, getmypid() . ' : ' . date('Y-m-d H:i:s') . " : {$num} \r\n"); flock($fp, LOCK_UN); echo getmypid(), ": success \r\n"; ++$num; } exit; } } foreach($pids as $k => $v) { if($v) { pcntl_waitpid($v, $status); } } fclose($fp);
# php fork.php         //結果 
57641: success 57641: success 57641: success 57641: success 57641: success 57642: success 57642: success 57642: success 57642: success 57642: success 57643: success 57643: success 57643: success 57643: success 57643: success 
//test.log 日志內容 
57641 : 2018-02-02 15:55:48 : 1 
57641 : 2018-02-02 15:55:48 : 2 
57641 : 2018-02-02 15:55:48 : 3 
57641 : 2018-02-02 15:55:48 : 4 
57641 : 2018-02-02 15:55:48 : 5 
57642 : 2018-02-02 15:55:48 : 1 
57642 : 2018-02-02 15:55:48 : 2 
57642 : 2018-02-02 15:55:48 : 3 
57642 : 2018-02-02 15:55:48 : 4 
57642 : 2018-02-02 15:55:48 : 5 
57643 : 2018-02-02 15:55:48 : 1 
57643 : 2018-02-02 15:55:48 : 2 
57643 : 2018-02-02 15:55:48 : 3 
57643 : 2018-02-02 15:55:48 : 4 
57643 : 2018-02-02 15:55:48 : 5 

我們可以看到三個子進程的pid,它們分別執行了5次,時間幾乎是在同時。但是$num的值並沒像我們期望的那樣從1-15進行遞增。子進程中的變量是各自獨立的,互不影響。子進程會自動復制父進程空間里的變量。

 

 

參考原文https://www.cnblogs.com/jkko123/p/6351690.html


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM