PHP多進(jìn)程編程之僵尸進(jìn)程問題的理解
PHP多進(jìn)程編程之僵尸進(jìn)程問題的理解
使用pcntl_fork函數(shù)可以讓PHP實(shí)現(xiàn)多進(jìn)程并發(fā)或者異步處理的效果:http://www.dbjr.com.cn/article/125789.htm
那么問題是我們產(chǎn)生的進(jìn)程需要去控制,而不能置之不理。最基本的方式就是fork進(jìn)程和殺死進(jìn)程。
通過利用pcntl_fork函數(shù),我們已經(jīng)有了新的子進(jìn)程,而子進(jìn)程接下來完成我們需要處理的內(nèi)容,那么我們就暫且叫做service()吧,而且我們需要很多個(gè)service()進(jìn)行處理,再次參照我們之前的需求,父進(jìn)程需要一直循環(huán)讀取配置文件,等待文件發(fā)生改變。通過對pcntl_fork的方式,很容易我們就可以寫出如下代碼:
$res = config(); //kill進(jìn)程 for($i = 0; $i < $res[sum]; $i++) { $pid = pcntl_fork(); if ($pid == 0) { service(); return; } }
代碼中注釋的地方我們需要在配置文件中發(fā)生改變的時(shí)候殺死進(jìn)程,殺死進(jìn)程的的方式很簡單,可以使用kill命令直接殺死,比如(假設(shè)pid為123):
1 kill 123
但是我們發(fā)現(xiàn),使用這個(gè)殺死進(jìn)程的方式并沒有真正的把進(jìn)程殺死,這個(gè)子進(jìn)程被殺死后還占用這個(gè)進(jìn)程的資源,我們成為僵尸進(jìn)程,僵尸進(jìn)程是使用kill命令無法殺死的。想要解決這個(gè)問題,我們能做的只有兩種方式。
1. shutdown
2. 殺死該進(jìn)程的父進(jìn)程。
但是這兩種方法都不行,因?yàn)檫@個(gè)程序的目的是監(jiān)控常駐在服務(wù)器內(nèi),服務(wù)器不能關(guān)閉,并且父進(jìn)程也不能被干掉。這時(shí)候我們看到了官方文檔對于fork方法的解釋:
pcntl_wait($status); //等待子進(jìn)程中斷,防止子進(jìn)程成為僵尸進(jìn)程。
原來有種方式可以防止進(jìn)程成為僵尸進(jìn)程,但是,官網(wǎng)給出的代碼是這樣子的:
$pid = pcntl_fork(); //父進(jìn)程和子進(jìn)程都會執(zhí)行下面代碼 if ($pid == -1) { //錯(cuò)誤處理:創(chuàng)建子進(jìn)程失敗時(shí)返回-1. die('could not fork'); } else if ($pid) { //父進(jìn)程會得到子進(jìn)程號,所以這里是父進(jìn)程執(zhí)行的邏輯 pcntl_wait($status); //等待子進(jìn)程中斷,防止子進(jìn)程成為僵尸進(jìn)程。 } else { //子進(jìn)程得到的$pid為0, 所以這里是子進(jìn)程執(zhí)行的邏輯。 }
什么意思呢?就是父進(jìn)程會等待子進(jìn)程運(yùn)行,等子進(jìn)程運(yùn)行結(jié)束之后,才會進(jìn)行下一步,并且也會消除僵尸進(jìn)程。但是這里又和我們的需求不符合了,我們的子進(jìn)程為一個(gè)死循環(huán)的程序,不斷的查找輸出,更本沒有結(jié)束的時(shí)候,并且我們需要的是異步處理而不是同步。但是這個(gè)方法可以用嗎?其實(shí)當(dāng)然可以。
在pcntl_wait的文檔中是這么解釋這個(gè)函數(shù)的:
wait函數(shù)刮起當(dāng)前進(jìn)程的執(zhí)行直到一個(gè)子進(jìn)程退出或接收到一個(gè)信號要求中斷當(dāng)前進(jìn)程或調(diào)用一個(gè)信號處理函數(shù)。 如果一個(gè)子進(jìn)程在調(diào)用此函數(shù)時(shí)已經(jīng)退出(俗稱僵尸進(jìn)程),此函數(shù)立刻返回。子進(jìn)程使用的所有系統(tǒng)資源將 被釋放。關(guān)于wait在您系統(tǒng)上工作的詳細(xì)規(guī)范請查看您系統(tǒng)的wait(2)手冊。
我們發(fā)現(xiàn),當(dāng)這個(gè)函數(shù)發(fā)現(xiàn)子進(jìn)程成為了僵尸進(jìn)程就會釋放僵尸進(jìn)程的資源——前提是這個(gè)僵尸進(jìn)程為這個(gè)父進(jìn)程的子進(jìn)程。那么我們就可以巧妙的利用這個(gè)方式讓這些僵尸進(jìn)程釋放資源了,所以就有了如下代碼:
posix_kill(123, 9); pcntl_wait($status);
這樣我們先使用kill干掉這個(gè)進(jìn)程,這個(gè)進(jìn)程就不會再運(yùn)行了,但是這個(gè)進(jìn)程成為了僵尸進(jìn)程,占用著資源,我們下一句就執(zhí)行一次pcntl_wait()讓這些僵尸進(jìn)程釋放資源,這樣,子進(jìn)程才真正的被終止了,僵尸進(jìn)程被消除了。
如有疑問請留言或者到本站社區(qū)交流討論,感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
- PHP如何限制定時(shí)任務(wù)的進(jìn)程數(shù)量
- php多進(jìn)程中的阻塞與非阻塞操作實(shí)例分析
- php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析
- PHP多進(jìn)程之pcntl_fork的實(shí)例詳解
- PHP基于文件鎖解決多進(jìn)程同時(shí)讀寫一個(gè)文件問題示例
- PHP守護(hù)進(jìn)程的兩種常見實(shí)現(xiàn)方式詳解
- php中實(shí)現(xiàn)進(jìn)程鎖與多進(jìn)程的方法
- PHP多進(jìn)程編程總結(jié)(推薦)
- 詳解PHP調(diào)用Go服務(wù)的正確方式
相關(guān)文章
php mysql實(shí)現(xiàn)mysql_select_db選擇數(shù)據(jù)庫
在PHP中,與MySQL服務(wù)器建立連接后,需要確定所要連接的數(shù)據(jù)庫,此時(shí)我們可以使用mysql_select_db函數(shù),該函數(shù)用于選擇需要操作的數(shù)據(jù)庫,需要的朋友可以參考下2016-12-12PHP設(shè)計(jì)模式之工廠方法設(shè)計(jì)模式實(shí)例分析
這篇文章主要介紹了PHP設(shè)計(jì)模式之工廠方法設(shè)計(jì)模式,結(jié)合實(shí)例形式分析了工廠方法設(shè)計(jì)模式的概念、原理以及php一般工廠方法模式與參數(shù)化工廠方法模式具體實(shí)現(xiàn)技巧,需要的朋友可以參考下2018-04-04php讀取圖片內(nèi)容并輸出到瀏覽器的實(shí)現(xiàn)代碼
如果php以圖片,zip,exe等文件輸出到瀏覽器,而前面還輸出了其他字符,那就會是你看到的亂碼2013-08-08PHP htmlspecialchars_decode()函數(shù)用法講解
今天小編就為大家分享一篇關(guān)于PHP htmlspecialchars_decode()函數(shù)用法講解,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-03-03