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

來源:互聯網
上載者:User
這篇文章主要介紹了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個了。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.