首頁  >  文章  >  後端開發  >  php多進程中關於pcntl_fork的詳細介紹

php多進程中關於pcntl_fork的詳細介紹

黄舟
黄舟原創
2017-10-16 09:19:131977瀏覽

這篇文章主要介紹了PHP多進程編之pcntl_fork的實例詳解的相關資料,希望透過本文能幫助到大家,讓大家理解掌握這部分內容,需要的朋友可以參考下

#PHP多進程編之pcntl_fork的實例詳解

其實PHP是支援並發的,只是平常很少使用而已。平時使用最多的應該是使用PHP-FMP調度php進程了吧。

但是,PHP的使用不限於做Web,我們完全也可以使用PHP來進行系統工具類的編程,做監控或是維運。在使用這些方向的時候,我們可以使用到PHP的更多特性,例如並發(多進程)、socket程式設計等。

那麼接下來就來談談我遇到的PHP多進程的程式設計。這個多進程的使用是有一個背景的,下面模糊描述一下背景。

我需要一個監控系統,當然使用PHP語言,監控系統需要監控很多種系統指標,為了讓每個監控指標之間盡量專心的去做自己的事情,就需要單獨使用一個進程去監控一個指標,還有一個行程去讀取配置,拿到配置之後,依照配置去啟動每個行程。

那麼,這就需要我所說的多進程了。

  1. 先啟動一個主進程,主進程用來讀取組態資訊。例如,我讀取到了我需要監控5個指標

  2. 接下來主進程啟動5個子進程,分別監控這5個指標。

  3. 建立好5個指標監控進程之後之後,主進程進行監聽設定。

  4. 一旦配置改變,殺死先前的進程並重新建立進程。

比相對來說比較清晰的邏輯。那接下來我們就化簡一下操作:簡單的說就是一個主進程創建5個子進程。

首先,建立一個行程在需要使用php的一個函式pcntl_fork(),這個函式可能有的同學不太熟,不過接觸過Linux C變成的人都知道Linux下有個叫fork()的函數,用來建立子進程。這個函數和Linux下這個函數是一個意思。要注意的是,這個函數在Linux下才能使用,而且需要安裝pcntl的擴充功能。

對於這個函數怎麼使用,我們可以查閱官方文件:http://php.net/manual/zh/function.pcntl-fork.php

官方文件是這樣說的:

pcntl_fork()函數建立一個子進程,這個子進程僅PID(進程號) 和PPID(父進程號)與其父進程不同。 fork如何在您的系統工作的詳細資訊請查閱您的系統 的fork(2)手冊。

成功時,在父行程執行緒內傳回產生的子程序的PID,在子程序執行緒內回傳0。失敗時,在 父進程上下文傳回-1,不會建立子進程,並且會引發一個PHP錯誤。

這樣就可以建立一個子行程了,子行程建立成功以後會執行pcntl_fork()之後的方法。那麼對於這個函數的回傳值我們要如何理解呢?

是這樣的,我們呼叫函數創建進程的時候,函數執行時有時間的,而新的進程剛好是在函數執行開始和結束之間創建出來的,這樣,新的進程也執行了這個函數,所以函數也需要有回傳值。那麼對於該函數一次執行之後,父進程和子進程都會受到該函數的回傳值,由於父進程創建了子進程,而子進程並沒有創建新的進程,所以子進程對於這個函數的返回結果是沒有的,所以就給他賦了一個0。而父進程創建了子進程,子進程是存在pid的,所以就得到了那個進程的pid。

我們可以寫個程式來了解:


$pid = pcntl_fork();
var_dump($pid);

這個呼叫會輸出兩個值,但我們如果直接print的只能看到一個值,也就是子程序的pid,但是使用var_dump我們就可以看到兩個值,是0和子程序的pid。 0這個值就是子進程回傳過來的。

那麼如何創建進程了解清楚之後,就可以開始創建進程了,我們需要創建5個進程,那麼我就循環5次創建進程。得到如下程式碼:


 $i=0;
 while($i!=5){
  $pid = pcntl_fork();
  echo $pid."---------hahah".$i++.PHP_EOL;
 }

這樣就寫好了,那麼執行一下吧。啊?發現不是5個進程啊,發現有好多個進程,而且最後一個hahah4這個輸出有32個,為什麼是32呢?我們算一算。 2^5=32,為什麼最後的執行緒數以指數成長了呢?

想發現這個並不難,因為我們之後的每一條都執行了while循環,到最後成了進程的指數增長——也就是說fork的時候把while循環也帶了進去。但是我們只是要5個進程而已。怎麼辦呢?

透過先前對函數的研究可以看到,子進程中會回傳一個為0的值,那麼我們就可以知道,0為子進程的標記。我們可以透過對子進程標記來結束進程執行。所以我們可以將我們的程式碼修改為如下形式:


#
$i=0;
while($i!=5){
 $pid = pcntl_fork();
 echo $pid."---------hahah".$i++.PHP_EOL;
 if ($pid == 0) {
  echo "子进程".PHP_EOL;
  return;
 }
}

因为0其实是对子进程的标记,那么pid这个变量在子进程里实际上是0的,所以当发现pid的值为0的时候,我们就可以断定我们当前进程为一个子进程,不需要在让他执行while并创建子进程的子进程了,所以在执行完我们的内容之后就return或者exit退出这个执行就好了。这样就能保证我们执行创建了5个进程而不是32个了。

以上是php多進程中關於pcntl_fork的詳細介紹的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn