Linux下多進(jìn)程編程API

進(jìn)程是Linux操作系統(tǒng)環(huán)境的基礎(chǔ),它控制著系統(tǒng)幾乎所有的活動(dòng),下面介紹Linux下多進(jìn)程的系統(tǒng)調(diào)用API爬早。

fork()系統(tǒng)調(diào)用

Linux下創(chuàng)建新進(jìn)程的系統(tǒng)調(diào)用時(shí)fork(),定義如下:

#include <sys/types.h>
#include <unistd.h>

pid_t fork(void);

該函數(shù)每的每次調(diào)用都返回兩次,在父進(jìn)程中返回的是子進(jìn)程的PID, 在子進(jìn)程中返回0 臣樱。返回值是后續(xù)用來判斷當(dāng)前進(jìn)程為父進(jìn)程還是子進(jìn)程的依據(jù)。fork調(diào)用失敗返回-1腮考,并設(shè)置errno雇毫。
fork()函數(shù)復(fù)制當(dāng)前進(jìn)程,在內(nèi)核進(jìn)程表中創(chuàng)建一個(gè)新的進(jìn)程表項(xiàng)踩蔚。新的進(jìn)程表項(xiàng)有許多屬性和原進(jìn)程相同棚放。比如,堆指針馅闽,棧指針飘蚯,標(biāo)志寄存器的值馍迄。但也有許多屬性被賦予了新的值,比如孝冒。PPID柬姚,信號(hào)位圖被清除,原來進(jìn)程設(shè)置的信號(hào)處理函數(shù)不在對新進(jìn)程起作用庄涡。
子進(jìn)程的代碼和父進(jìn)程的代碼完全相同,同時(shí)他還會(huì)復(fù)制父進(jìn)程的數(shù)據(jù)(堆數(shù)據(jù)搬设,棧數(shù)據(jù)和靜態(tài)數(shù)據(jù))穴店。但是數(shù)據(jù)的復(fù)制采用的是寫時(shí)復(fù)制(copy on write)。即只有在任一進(jìn)程對數(shù)據(jù)執(zhí)行了寫操作的時(shí)候拿穴,復(fù)制才會(huì)發(fā)生(先是發(fā)生缺頁中斷泣洞,然后操作系統(tǒng)給子進(jìn)程分配內(nèi)存并復(fù)制父進(jìn)程的數(shù)據(jù))。即便如此默色,我們在程序中分配了大量內(nèi)存的時(shí)候球凰,也要謹(jǐn)慎使用fork(),避免復(fù)制沒有必要的內(nèi)存和數(shù)據(jù)腿宰。
此外呕诉,父進(jìn)程中打開了文件描述符,fork()后吃度,子進(jìn)程也是打開的甩挫,且文件描述符的引用計(jì)數(shù)加1。不僅如此椿每,父進(jìn)程的用戶根目錄伊者,當(dāng)前工作目錄等變量的引用計(jì)數(shù)都會(huì)加1。

exec系列系統(tǒng)調(diào)用

有時(shí)我們需要在子進(jìn)程中執(zhí)行其他程序间护,即在fork()后替換當(dāng)前進(jìn)程的映像亦渗,需要使用到一下的函數(shù):

extern char** environ;                                                                   
// 替換當(dāng)前進(jìn)程映像                                                                      
// path 參數(shù)指定可執(zhí)行文件的全路徑,                                                      
// arg 接受可變參數(shù)                                                                      
int execl(const char* path, const char* arg, ...);                                       
// file 參數(shù)可以接受文件名,該文件的具體位置則在環(huán)境變量PATH中搜索,                       
int execlp(const char* file, const char* arg, ...);                                      
// argp 用于設(shè)置環(huán)境變量                                                                 
int execle(const char* path, const char* arg, ..., char* const envp[]);                  
// argv 表示可以接受參數(shù)數(shù)組,他們都會(huì)被傳遞給新進(jìn)程                                      
int execv(const char* path, char* const argv[]);                                         
int execvp(const char* file, char* argv[]);                                              
int execve(const char* path, char* const argv[], char* const envp[]);    

一般情況下,exec函數(shù)是不返回的汁尺,除非出錯(cuò)法精,出錯(cuò)時(shí)返回 -1,并設(shè)置errno均函。
如果exec執(zhí)行成果亿虽,exec下面的代碼不會(huì)執(zhí)行的,類似于return 語句苞也。

exec 函數(shù)不會(huì)關(guān)閉源程序打開的文件描述符洛勉,除非該文件描述符被設(shè)置了類似于SOCK_CLOEXEC的屬性。

wait處理僵尸進(jìn)程

在多進(jìn)程的編程中如迟,父進(jìn)程一般會(huì)跟蹤子進(jìn)程的退出狀態(tài)收毫。因此攻走,當(dāng)子進(jìn)程結(jié)束運(yùn)行時(shí),內(nèi)核不會(huì)立即釋放該進(jìn)程的進(jìn)程表表項(xiàng)此再,以滿足父進(jìn)程后續(xù)對子進(jìn)程進(jìn)程退出信息的查詢昔搂。

在子進(jìn)程退出,父進(jìn)程沒有獲取其退出狀態(tài)之前输拇,我們?nèi)蝿?wù)他是僵尸進(jìn)程摘符。

在僵尸態(tài)的進(jìn)程,它依然占據(jù)著內(nèi)核資源策吠。這時(shí)絕對不允許的逛裤,畢竟內(nèi)核資源有限。

                                                                                         
#include <sys/types.h>                                                                   
#include <sys/wait.h>                                                                    
                                                                                         
// 返回子進(jìn)程的pid, stat_loc獲取退出狀態(tài) 猴抹,阻塞等待带族。                                                
pid_t wait(int* stat_loc);             
// pid == -1時(shí),獲取任意個(gè)子進(jìn)程蟀给,非阻塞的蝙砌。                                              
pid_t waitpid(pid_t pid, int* stat_loc, int option);                                     

wait函數(shù)阻塞進(jìn)程,直到該進(jìn)程的某個(gè)子進(jìn)程運(yùn)行結(jié)束為止跋理。
waitpid函數(shù)只等待pid參數(shù)指定的子進(jìn)程择克。如果pid取值為-1,那么和wait函數(shù)相同薪介。waitpid是非阻塞的祠饺,如果pid指定的目標(biāo)子進(jìn)程還沒有結(jié)束,或者意外終止汁政,waitpid 返回0道偷,如果子進(jìn)程確實(shí)正確退出了,waitpid返回子進(jìn)程的PID记劈。waitpid調(diào)用失敗返回-1,并設(shè)置errno勺鸦。

static void handle_chile(int signal) {
   pid_t pid;
   int stat ;
   while ( (pid = waitpid(-1, &stat, WNOHANG)) > 0) {
        // 對子進(jìn)程進(jìn)行善后處理 
  }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市目木,隨后出現(xiàn)的幾起案子换途,更是在濱河造成了極大的恐慌,老刑警劉巖刽射,帶你破解...
    沈念sama閱讀 216,324評(píng)論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件军拟,死亡現(xiàn)場離奇詭異,居然都是意外死亡誓禁,警方通過查閱死者的電腦和手機(jī)懈息,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來摹恰,“玉大人辫继,你說我怎么就攤上這事怒见。” “怎么了姑宽?”我有些...
    開封第一講書人閱讀 162,328評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵遣耍,是天一觀的道長。 經(jīng)常有香客問我炮车,道長舵变,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,147評(píng)論 1 292
  • 正文 為了忘掉前任瘦穆,我火速辦了婚禮棋傍,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘难审。我一直安慰自己,他們只是感情好亿絮,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,160評(píng)論 6 388
  • 文/花漫 我一把揭開白布告喊。 她就那樣靜靜地躺著,像睡著了一般派昧。 火紅的嫁衣襯著肌膚如雪黔姜。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,115評(píng)論 1 296
  • 那天蒂萎,我揣著相機(jī)與錄音秆吵,去河邊找鬼。 笑死五慈,一個(gè)胖子當(dāng)著我的面吹牛纳寂,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播泻拦,決...
    沈念sama閱讀 40,025評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼毙芜,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了争拐?” 一聲冷哼從身側(cè)響起腋粥,我...
    開封第一講書人閱讀 38,867評(píng)論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎架曹,沒想到半個(gè)月后隘冲,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,307評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡绑雄,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,528評(píng)論 2 332
  • 正文 我和宋清朗相戀三年展辞,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片绳慎。...
    茶點(diǎn)故事閱讀 39,688評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡纵竖,死狀恐怖漠烧,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情靡砌,我是刑警寧澤已脓,帶...
    沈念sama閱讀 35,409評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站通殃,受9級(jí)特大地震影響度液,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜画舌,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,001評(píng)論 3 325
  • 文/蒙蒙 一堕担、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧曲聂,春花似錦霹购、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至旭咽,卻和暖如春贞奋,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背穷绵。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評(píng)論 1 268
  • 我被黑心中介騙來泰國打工轿塔, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人仲墨。 一個(gè)月前我還...
    沈念sama閱讀 47,685評(píng)論 2 368
  • 正文 我出身青樓勾缭,卻偏偏與公主長得像,于是被迫代替她去往敵國和親宗收。 傳聞我的和親對象是個(gè)殘疾皇子漫拭,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,573評(píng)論 2 353

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