PHP FPM源代碼反芻品味之一:無(wú)限運(yùn)行程序

基礎(chǔ): 無(wú)限運(yùn)行程序.

程序可以簡(jiǎn)單的分為兩種類(lèi)型:

  1. 程序啟動(dòng)后,一段時(shí)間后,完成了任務(wù),會(huì)主動(dòng)退出,這里稱(chēng)為有限程序.
  2. 程序啟動(dòng)后,會(huì)一直運(yùn)行,不會(huì)主動(dòng)退出,這里稱(chēng)為無(wú)限程序.

顯然,服務(wù)器程序如nginx,php-fpm和桌面圖形界面(GUI)程序如firefox,word 都屬無(wú)限程序.
這類(lèi)程序的共同點(diǎn)是啟動(dòng)后一直運(yùn)行,如果不出故障,能源足夠,且未收到退出指令的情況下,無(wú)限程序會(huì)永久運(yùn)行.

服務(wù)器程序和圖形界面,一個(gè)前端,一個(gè)后端,相距甚遠(yuǎn). 兩者內(nèi)部的核心機(jī)制且及其相近.
服務(wù)器程序,啟動(dòng)后,等待網(wǎng)絡(luò)請(qǐng)求, 并做出響應(yīng).
圖形界面,啟動(dòng)后,等待用戶(hù)事件,點(diǎn)擊或鍵盤(pán)輸入等,并做出響應(yīng)
簡(jiǎn)單的道理,如果程序運(yùn)行一段時(shí)間后就退出, 就無(wú)法響應(yīng)網(wǎng)絡(luò)請(qǐng)求或用戶(hù)事件.

我們簡(jiǎn)單的想一下,一個(gè)程序要如何才能無(wú)限運(yùn)行呢? 再?gòu)?fù)雜的邏輯,總有計(jì)算完成的時(shí)候.
直覺(jué)告訴我們,程序要無(wú)限運(yùn)行,里面應(yīng)該有個(gè)無(wú)限循環(huán).
沒(méi)錯(cuò)服務(wù)器程序和圖形界面程序都有個(gè)無(wú)限循環(huán).

程序在無(wú)限循環(huán)里運(yùn)行, 如果馬不停蹄的飛轉(zhuǎn), 會(huì)獨(dú)占100%CPU資源,機(jī)器不累,看的人都累.
于是我們想到,應(yīng)該在程序無(wú)限循環(huán)里歇一歇,最簡(jiǎn)單的辦法就是sleep一下:如下:

while(1){
  process();
  sleep(1);
}

這樣程序可以一直運(yùn)行且不獨(dú)占CPU,很多人都這樣做過(guò).
但在正式的產(chǎn)品程序里,這不是個(gè)好辦法.
首先, sleep時(shí)間太短,沒(méi)意義,sleep時(shí)間太長(zhǎng)會(huì)導(dǎo)致響應(yīng)緩慢停頓.
還有,sleep 效率也不高.

怎么辦?

多數(shù)服務(wù)器程序和圖形界面程序在無(wú)限循環(huán)里,
通常會(huì)在對(duì)一個(gè)對(duì)文件句柄(網(wǎng)絡(luò)連接也是文件)的操作上停一下,收到信息后或超時(shí)后繼續(xù)循環(huán)運(yùn)行.

程序停在那里等待信息,有個(gè)專(zhuān)業(yè)術(shù)語(yǔ)叫阻塞(blocking).
操作系統(tǒng)在內(nèi)核層面上,支持阻塞(blocking)機(jī)制, 所以程序阻塞在對(duì)IO的操作上,高效且占用的資源很小.

有的程序員說(shuō),我開(kāi)發(fā)了很多android應(yīng)用,沒(méi)看到哪里有無(wú)限循環(huán)啊?
主要的原因就android在框架里實(shí)現(xiàn)了,對(duì)一般開(kāi)發(fā)者不可見(jiàn),如果你看android框架源碼,會(huì)在某個(gè)地方發(fā)現(xiàn)無(wú)限循環(huán).

在服務(wù)器程序和圖形界面程序的另一個(gè)很相似的地方就是:事件和事件隊(duì)列.
通常與事件相關(guān)的詳細(xì)信息和回調(diào)函數(shù)會(huì)包裝成一個(gè)事件對(duì)象,放到事件隊(duì)列里. (C語(yǔ)言用結(jié)構(gòu)體表示對(duì)象)
對(duì)文件句柄fd阻塞讀取操作(監(jiān)聽(tīng))只是個(gè)獲取個(gè)觸發(fā)信號(hào).
無(wú)限循環(huán)里,獲取和處理隊(duì)列里的事件.
常見(jiàn)的設(shè)計(jì)中,定時(shí)運(yùn)行邏輯,會(huì)放到一個(gè)定時(shí)隊(duì)列里,無(wú)限循環(huán)時(shí)順帶檢查定時(shí)隊(duì)列,處理到時(shí)的運(yùn)行邏輯.

以下用一個(gè)簡(jiǎn)單的程序說(shuō)明:

// simple_event.c
#include <stdio.h>

int  fd= 0;
char event[100];

void wait_event(){
    int length;
    printf("Please input event\n");
    length = read(fd,event,sizeof(event));
    event[length] = 0;
    printf("Recieved event: %s",event);
}

void process_event(){
     printf("Processed event: %s",event);
}

int main() {
    while(1){
        wait_event();
        process_event();
    }
    return 0;
}

運(yùn)行

gcc -o  simple_event simple_event.c
./simple_event
Please input event
keyup
Recieved event: keyup
Processed event: keyup
Please input event
click
Recieved event: click
Processed event: click

這個(gè)很簡(jiǎn)單的無(wú)限程序,確也體現(xiàn)了服務(wù)器和圖形界面程序的基礎(chǔ)結(jié)構(gòu):
無(wú)限循環(huán),阻塞監(jiān)聽(tīng)事件, 處理事件.
這個(gè)程序占用CPU很小,也說(shuō)明了阻塞操作在無(wú)限循環(huán)里的重要性.
這個(gè)程序和成品無(wú)限程序相比,還需要改進(jìn):

  1. 這個(gè)程序監(jiān)聽(tīng)0號(hào)文件(fd=0), 也就是標(biāo)準(zhǔn)輸入鍵盤(pán), 成品程序通常會(huì)使用pipe或sockpair 創(chuàng)建虛擬文件用于通信. (重復(fù)一下,unix下,一切IO設(shè)備皆文件)
  2. 成品程序通常會(huì)在循環(huán)中加入處理定時(shí)任務(wù)的邏輯.
  3. 沒(méi)有輸入的情況下,這個(gè)程序會(huì)一直阻塞, 無(wú)法處理其他任務(wù).

成品程序通常會(huì)搭配select (epoll) 使用, 設(shè)置阻塞超時(shí)時(shí)間,以便處理到時(shí)的定時(shí)任務(wù).
這里的阻塞超時(shí)時(shí)間,通常依據(jù)最近的定時(shí)任務(wù)的時(shí)間來(lái)設(shè)定。
改良后装哆,循環(huán)的大致結(jié)構(gòu)如下:

while(1){
    timeout = get_timeout(timer_task_queue); 依據(jù)最近的定時(shí)任務(wù)
    ready_io_events = wait_io_event(timeout);
    process(ready_io_events);
    process_timer_task()
}

總結(jié):

無(wú)線循環(huán)里:
1,獲取阻塞超時(shí)時(shí)間(依據(jù)下一個(gè)定時(shí)任務(wù))靡羡,
2,阻塞等待IO事件援奢,
3驻债,處理IO事件挚躯,
4强衡,處理定時(shí)任務(wù)。
周而復(fù)始码荔,不停運(yùn)行漩勤,這可以說(shuō)是服務(wù)器和圖形界面程序設(shè)計(jì)的一個(gè)套路。
了解了這個(gè)套路缩搅,研究無(wú)限程序源代碼就不會(huì)陌生越败。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市硼瓣,隨后出現(xiàn)的幾起案子究飞,更是在濱河造成了極大的恐慌,老刑警劉巖巨双,帶你破解...
    沈念sama閱讀 217,406評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件噪猾,死亡現(xiàn)場(chǎng)離奇詭異霉祸,居然都是意外死亡筑累,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門(mén)丝蹭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)慢宗,“玉大人,你說(shuō)我怎么就攤上這事奔穿【倒粒” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,711評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵贱田,是天一觀的道長(zhǎng)缅茉。 經(jīng)常有香客問(wèn)我,道長(zhǎng)男摧,這世上最難降的妖魔是什么蔬墩? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,380評(píng)論 1 293
  • 正文 為了忘掉前任译打,我火速辦了婚禮,結(jié)果婚禮上拇颅,老公的妹妹穿的比我還像新娘奏司。我一直安慰自己,他們只是感情好樟插,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布韵洋。 她就那樣靜靜地躺著,像睡著了一般黄锤。 火紅的嫁衣襯著肌膚如雪搪缨。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,301評(píng)論 1 301
  • 那天猜扮,我揣著相機(jī)與錄音勉吻,去河邊找鬼。 笑死旅赢,一個(gè)胖子當(dāng)著我的面吹牛齿桃,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播煮盼,決...
    沈念sama閱讀 40,145評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼短纵,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了僵控?” 一聲冷哼從身側(cè)響起香到,我...
    開(kāi)封第一講書(shū)人閱讀 39,008評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎报破,沒(méi)想到半個(gè)月后悠就,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,443評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡充易,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評(píng)論 3 334
  • 正文 我和宋清朗相戀三年梗脾,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片盹靴。...
    茶點(diǎn)故事閱讀 39,795評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡炸茧,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出稿静,到底是詐尸還是另有隱情梭冠,我是刑警寧澤,帶...
    沈念sama閱讀 35,501評(píng)論 5 345
  • 正文 年R本政府宣布改备,位于F島的核電站控漠,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏悬钳。R本人自食惡果不足惜盐捷,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評(píng)論 3 328
  • 文/蒙蒙 一柬脸、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧毙驯,春花似錦倒堕、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,731評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至铭段,卻和暖如春骤宣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背序愚。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,865評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工憔披, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人爸吮。 一個(gè)月前我還...
    沈念sama閱讀 47,899評(píng)論 2 370
  • 正文 我出身青樓芬膝,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親形娇。 傳聞我的和親對(duì)象是個(gè)殘疾皇子锰霜,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容