首先需要解釋的是什麼是守護進程。
守護進程就是在後台一直啟動並執行進程。比如我們啟動的httpd,mysqld等進程都是常駐記憶體內啟動並執行程式。
針對需求進行分析:
需求:有一個常駐隊列messageQueue(假設在redis記憶體中),這個隊列會有可能有請求不週期性往隊列中增加元素。同時我們要求在隊列中有元素的時候,按照隊列順序將元素pop出來,並進行處理(假設這個處理只是echo ‘test');
解決方案:
現在假設已經有了兩個函數
function oPopMessageQueue(){ …} //擷取隊列最後一個元素;
function vDealElement($element) { …} 處理元素;
要求寫出一個精靈,完成上面的需求。
程式:
好了,這個程式很容易想到,可以使用while迴圈來做
複製代碼 代碼如下:while(true)
{
if( $element = oPopMessageQueue())
{
vDealElement($element);
}
}
考慮1 : 這個程式如果一直跑的話已經可以滿足上面的需求了.
但是考慮到:1 用php進程跑有可能會由於各種情況(比如已耗用時間過長),進程掛了,這樣程式就無法自動重連了.
方法:使用cron
我們在定時指令碼中每10分鐘起一個進程跑這個程式。
然後設定這個程式的已耗用時間為10分鐘,10分鐘後自動取消,於是代碼變成
複製代碼 代碼如下:while(true)
{
if($element = oPopMessageQueue())
{
vCheckTimeLimit();
vDealElement($elemnt);
}
}
$timeStart = 0;
function vCheckTimeLimit()
{
global $timeStart;
if(empty($timeStart))
{
$timeStart = time();
}
if(time() - $timeStart > 60*10)
{
exit;
}
}
考慮2,可能會有這種需求: 需要有隨時讓指令碼暫停功能:
於是考慮使用檔案來增加暫停功能複製代碼 代碼如下:while(true)
{
if($element = oPopMessageQueue())
{
vCheckTimeLimit();
vCheckEnd();
vDealElement($elemnt);
}
}
function vCheckEnd()
{
if(file_exists("/home/JesephYe/end"))
{
exit;
}
}
考慮3, 是否可以改成多線程的程式,讓啟動並執行效率更高?
這個只要把cron的10分鐘起一個進程的限制改成每1分鐘起一個進程就好了
這樣能保證有10個線程在運行程式
但是有一個基本要求是:oPopMessageQueue()是一個原子操作