來源:不言 發(fā)布時間:2018-12-13 15:29:36 閱讀量:1241
本篇文章給大家?guī)淼膬?nèi)容是關(guān)于PHPSocket編程中多進程的回聲服務(wù)器的詳細(xì)介紹(圖文) ,有一定的參考價值,有需要的朋友可以參考一下,希望對你有所幫助。
上次的回聲服務(wù)程序有個很大的缺點,就是只能同時連接一個客戶端,這明顯是不合理的。
所以這次采用多進程的方式來實現(xiàn)同時為多個客戶端提供服務(wù)。
以下是最終的效果:
(通過 ps 命令可以查看到我們創(chuàng)建的工作進程)
(通過 xshell 打開多個終端,并向服務(wù)程序連接多個客戶端)
在開始編碼之前,先介紹一下什么是進程。
進程,即正在運行的占用內(nèi)存的程序,是一個獨立的內(nèi)存空間。比如我們在 windows 上打開記事本軟件,這個操作就相當(dāng)于打開了一個進程。
而多進程則是通過創(chuàng)建多個進程來共同完成一件事。
我們本次只需實現(xiàn)服務(wù)端的程序就行了,客戶端還使用之前的。沒讀過之前文章的可以查看歷史文章。
在PHP中,我們可以使用 pcntl_fork 函數(shù)來實現(xiàn)創(chuàng)建進程。
下面是函數(shù)的原型:
1 |
|
官方解釋:
成功時,在父進程執(zhí)行線程內(nèi)返回產(chǎn)生的子進程的PID,在子進程執(zhí)行線程內(nèi)返回0。失敗時,在 父進程上下文返回-1,不會創(chuàng)建子進程,并且會引發(fā)一個PHP錯誤。
為什么說是父進程執(zhí)行的線程呢,因為一個進程中至少包含一個線程 ,而這個線程則是進程的主線程。
父進程在調(diào)用 pcntl_fork 時,同時復(fù)制出一個獨立的子進程,這個子進程具有父進程同樣的上下文。也就是說兩個進程共享一個代碼而已。
下面進入編碼環(huán)節(jié):
通過執(zhí)行 pcntl_fork 函數(shù),同時復(fù)制了一個子進程,此時,如果上下文是父進程的執(zhí)行環(huán)境,則返回值為子進程的進程號。如果是子進程的執(zhí)行環(huán)境,則返回0。所以下面的if程序結(jié)構(gòu),兩個分支都執(zhí)行了。
子進程同樣還是以前的邏輯,用來接收客戶端的消息,同時發(fā)送給客戶端。
倒數(shù)第二行調(diào)用了 pcntl_waitpid 函數(shù),在講解這個函數(shù)之前我們先了解一下什么是僵尸進程。
正常情況下,子進程是通過父進程創(chuàng)建的。由于進程是互相獨立的內(nèi)存結(jié)構(gòu),所以父進程是不會知道子進程的運行狀態(tài)的。子進程完成自己的任務(wù)之后,并不能自己退出,這個時候需要父進程通過操作系統(tǒng)來取得子進程的狀態(tài),從而回收子進程。否則,我們的子進程將會成為一個垃圾資源,也就是僵尸進程。
下面是這個函數(shù)的函數(shù)原型:
1 |
|
官方解釋:
等待或返回fork的子進程狀態(tài)。
其實上面的代碼是有問題的,在 foreach 中第一次調(diào)用 pcntl_waitpid 之后 ,主進程其實此時是被阻塞著,一直在等待第一個子進程退出,而其他的子進程若此時異常退出,則并沒有被主進程回收,也就產(chǎn)生了僵尸進程。而在實際的開發(fā)中主進程還是要做其他的事情的。
所以這里推薦采用非阻塞的方式,很簡單,只需加上第三個參數(shù):WNOHANG。即:pcntl_waitpid ($pid, $status, WNOHANG)
這樣子可以在沒有子進程退出的情況下立刻返回,從而繼續(xù)執(zhí)行后續(xù)代碼。
相關(guān)推薦:
服務(wù)器端PHP多進程編程實戰(zhàn)_PHP教程