新聞中心
在Linux系統(tǒng)中,每個(gè)進(jìn)程都有一個(gè)唯一的進(jìn)程ID(PID),用來標(biāo)識(shí)這個(gè)進(jìn)程。除了PID外,每個(gè)進(jìn)程還有一個(gè)父進(jìn)程PID,即它的父進(jìn)程的PID。在這篇文章中,我們將深入理解Linux中父進(jìn)程PID的實(shí)現(xiàn)。

為企業(yè)提供成都網(wǎng)站設(shè)計(jì)、網(wǎng)站建設(shè)、外貿(mào)網(wǎng)站建設(shè)、網(wǎng)站優(yōu)化、營(yíng)銷型網(wǎng)站、競(jìng)價(jià)托管、品牌運(yùn)營(yíng)等營(yíng)銷獲客服務(wù)。創(chuàng)新互聯(lián)擁有網(wǎng)絡(luò)營(yíng)銷運(yùn)營(yíng)團(tuán)隊(duì),以豐富的互聯(lián)網(wǎng)營(yíng)銷經(jīng)驗(yàn)助力企業(yè)精準(zhǔn)獲客,真正落地解決中小企業(yè)營(yíng)銷獲客難題,做到“讓獲客更簡(jiǎn)單”。自創(chuàng)立至今,成功用技術(shù)實(shí)力解決了企業(yè)“網(wǎng)站建設(shè)、網(wǎng)絡(luò)品牌塑造、網(wǎng)絡(luò)營(yíng)銷”三大難題,同時(shí)降低了營(yíng)銷成本,提高了有效客戶轉(zhuǎn)化率,獲得了眾多企業(yè)客戶的高度認(rèn)可!
在Linux中,每個(gè)進(jìn)程都是由另一個(gè)進(jìn)程創(chuàng)建的,除了最頂級(jí)的進(jìn)程外,每個(gè)進(jìn)程都有一個(gè)父進(jìn)程。父進(jìn)程用fork()系統(tǒng)調(diào)用創(chuàng)建子進(jìn)程,fork()調(diào)用返回子進(jìn)程的PID給父進(jìn)程,父進(jìn)程使用這個(gè)PID來監(jiān)視子進(jìn)程的執(zhí)行。
在代碼實(shí)現(xiàn)中,一個(gè)進(jìn)程的PID和它的父進(jìn)程PID都是保存在進(jìn)程的task_struct數(shù)據(jù)結(jié)構(gòu)中的。這個(gè)數(shù)據(jù)結(jié)構(gòu)包含了進(jìn)程的所有信息,包括進(jìn)程的狀態(tài),進(jìn)程ID等等。在task_struct數(shù)據(jù)結(jié)構(gòu)中,父進(jìn)程PID被存儲(chǔ)在ppid成員中。
在一些情況下,進(jìn)程需要獲取它的父進(jìn)程PID。這個(gè)功能通常通過調(diào)用getppid()系統(tǒng)調(diào)用實(shí)現(xiàn)。getppid()返回進(jìn)程的父進(jìn)程PID。
另一種情況是在進(jìn)程需要向它的父進(jìn)程發(fā)送信號(hào)時(shí)。使用kill()系統(tǒng)調(diào)用和信號(hào)的名稱或數(shù)字,可以將一個(gè)信號(hào)發(fā)送給指定的進(jìn)程。如果向進(jìn)程的父進(jìn)程發(fā)送信號(hào),可以使用getppid()函數(shù)獲取父進(jìn)程PID,并將信號(hào)發(fā)送給這個(gè)PID。例如,下面的代碼將向父進(jìn)程發(fā)送一個(gè)SIGUSR1信號(hào):
pid_t parent_pid = getppid();
kill(parent_pid, SIGUSR1);
在Linux中,父進(jìn)程PID不僅僅用于進(jìn)程間通信。PID命名空間是進(jìn)程隔離的一個(gè)方面,它限制了進(jìn)程可以訪問的PID范圍。進(jìn)程只能看到在同一個(gè)PID命名空間中的PID。
在Linux中,每個(gè)進(jìn)程都有一個(gè)進(jìn)程樹。進(jìn)程樹是由init進(jìn)程(PID為1)創(chuàng)建的,init是所有進(jìn)程的祖先。在這個(gè)樹中,每個(gè)進(jìn)程都有一個(gè)父進(jìn)程和它的子進(jìn)程。通過獲取父進(jìn)程PID和子進(jìn)程PID,可以遍歷整個(gè)進(jìn)程樹。
父進(jìn)程PID在Linux中是非常重要的。通過使用它,進(jìn)程能夠與它的父進(jìn)程通信,獲取父進(jìn)程PID、發(fā)送信號(hào)等。在Linux的進(jìn)程樹中,每個(gè)進(jìn)程都有一個(gè)父進(jìn)程和唯一的PID,它們都保存在任務(wù)數(shù)據(jù)結(jié)構(gòu)中,為L(zhǎng)inux系統(tǒng)的進(jìn)程管理提供了強(qiáng)大的支持。
相關(guān)問題拓展閱讀:
- linux下的PID,PIDD是什么?他們之間的關(guān)系以及應(yīng)用是什么?
- linux中C語(yǔ)言關(guān)于進(jìn)程的創(chuàng)建
linux下的PID,PIDD是什么?他們之間的關(guān)系以及應(yīng)用是什么?
不知道你是不是學(xué)習(xí)編程的,如果不是就沒必要看蘆型改了。
1.PID是進(jìn)程標(biāo)識(shí)號(hào),它是一個(gè)進(jìn)程的唯一性標(biāo)識(shí)。PPID是該進(jìn)程父進(jìn)程的進(jìn)程標(biāo)識(shí)號(hào)。
2.fork和exec和pid完全就是2件事情不能混租罩為一談。fork是一個(gè)linux庫(kù)函數(shù)。他是用來創(chuàng)建一個(gè)新的進(jìn)程。至于exec是一個(gè)系陪判列函數(shù),C標(biāo)準(zhǔn)庫(kù)函數(shù),用來改變進(jìn)程上下文的。2者結(jié)合使用可以創(chuàng)建一個(gè)新的進(jìn)程。
3.如果創(chuàng)建新的進(jìn)程,一般是用fork,他會(huì)返回這個(gè)被創(chuàng)建進(jìn)程的PID,你可以通過PID找到這個(gè)進(jìn)程。
在 Linux 底下執(zhí)行一個(gè)指令時(shí),系統(tǒng)會(huì)給予這個(gè)動(dòng)作一個(gè) ID, 我們稱為 PID,而根據(jù)啟用這個(gè)指令的使用者與相關(guān)的指令功能,而給予這個(gè)特定好洞 PID 一組權(quán)限, 該指令可以進(jìn)行的行為則與這個(gè) PID 的權(quán)限有關(guān)。
linux進(jìn)程簡(jiǎn)介
Linux是一個(gè)多任歲州務(wù)的操作系統(tǒng),也就是說,在同一個(gè)時(shí)間內(nèi),可以有多個(gè)進(jìn)程同時(shí)執(zhí)行。如果讀者對(duì)計(jì)算機(jī)硬件體系有一定了解的話,會(huì)知道我們大家常用的單CPU計(jì)算機(jī)實(shí)際上在一個(gè)時(shí)間片斷內(nèi)只能執(zhí)行一條指令,那么Linux是如何實(shí)現(xiàn)多進(jìn)程同時(shí)執(zhí)行的呢?原來Linux使用了一種稱為”進(jìn)程調(diào)度(process scheng)”的手段,首先,為每個(gè)進(jìn)程指派一定的運(yùn)行時(shí)間,這個(gè)時(shí)間通常很短,短到以毫秒為單位,然后依照某種規(guī)則,從眾多進(jìn)程中挑選一個(gè)投入運(yùn)行,其他的進(jìn)程暫時(shí)等待,當(dāng)正在運(yùn)行的那個(gè)進(jìn)程時(shí)間耗盡,或執(zhí)行完畢退出,或因某種原因暫停,Linux就會(huì)重新進(jìn)行調(diào)度,挑選下一個(gè)進(jìn)程投入運(yùn)行。因?yàn)槊總€(gè)進(jìn)程占用的時(shí)間片都很短,在我們使用者的角度來看,就好像多個(gè)進(jìn)程同時(shí)運(yùn)行一樣了。
在Linux中,每個(gè)進(jìn)程在創(chuàng)建時(shí)都會(huì)被分配一個(gè)數(shù)據(jù)結(jié)構(gòu),稱為進(jìn)程控制塊(Process Control Block,簡(jiǎn)稱PCB)。PCB中包含了很多重要的信息,供系統(tǒng)調(diào)度和進(jìn)程本身執(zhí)行使用,其中最重要的莫過于進(jìn)程ID(process ID)了,進(jìn)程ID也被稱作進(jìn)程標(biāo)識(shí)符,是一個(gè)非負(fù)的整數(shù),在Linux操作系統(tǒng)中唯一地標(biāo)志一個(gè)進(jìn)程,在我們最常使用的I386架構(gòu)(即PC使用的架構(gòu))上,一個(gè)非負(fù)的整數(shù)的變化范圍是,這也是我們所有可能取到的進(jìn)程ID。其實(shí)從進(jìn)程ID的名字就可以看出,它就是進(jìn)程的身份證號(hào)碼,友雀枯每個(gè)人的身份證號(hào)碼都不會(huì)相同,每個(gè)進(jìn)程的進(jìn)程ID也不會(huì)相同。
一個(gè)或多個(gè)進(jìn)程可以合起來構(gòu)成一個(gè)進(jìn)程組(process group),一個(gè)或多個(gè)進(jìn)程組可以合起來構(gòu)成一個(gè)會(huì)話(session)。這樣我們就有了對(duì)進(jìn)程進(jìn)行批量操作的能力,比如通過向某個(gè)進(jìn)程組發(fā)送信號(hào)來實(shí)現(xiàn)向該組中的每個(gè)進(jìn)程發(fā)送信號(hào)。
最后,讓我們通過ps命令親眼看一看自己的系統(tǒng)中目前有多少進(jìn)程在運(yùn)行:
$ps -aux(以下是在我的計(jì)算機(jī)上的運(yùn)行結(jié)果,你的結(jié)果很可能與這不同。)
USERPID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
root.1 0.?S May15 0:04 init
root.0 0.?SW May15 0:00
root.0 0.?SW May15 0:00
root.0 0.?SWN May15 0:00
root.0 0.?SW May15 0:00
root.0 0.?SW May15 0:00
root.0 0.?SW May15 0:00
root.0 0.?SW May15 0:00
root.0 0.?SW /* 提供類型pid_t的定義 */
#include /* 提供函數(shù)的定義 */
pid_t getpid(void);
getpid的作用很簡(jiǎn)單,就是返回當(dāng)前進(jìn)程的進(jìn)程ID,請(qǐng)大家看以下的例子:
/* getpid_test.c */
#include
main()
{
printf(“The current process ID is %d
“,getpid());
}
細(xì)心的讀者可能注意到了,這個(gè)程序的定義里并沒有包含頭文件sys/types.h,這是因?yàn)槲覀冊(cè)诔绦蛑袥]有用到pid_t類型,pid_t類型即為進(jìn)程ID的類型。事實(shí)上,在i386架構(gòu)上(就是我們一般PC計(jì)算機(jī)的架構(gòu)),pid_t類型是和int類型完全兼容的,我們可以用處理整形數(shù)的方法去處理pid_t類型的數(shù)據(jù),比如,用”%d”把它打印出來。
編譯并運(yùn)行程序getpid_test.c:
$gcc getpid_test.c -o getpid_test
$./getpid_test
The current process ID is 1980
(你自己的運(yùn)行結(jié)果很可能與這個(gè)數(shù)字不一樣,這是很正常的。)
再運(yùn)行一遍:
$./getpid_test
The current process ID is 1981
正如我們所見,盡管是同一個(gè)應(yīng)用程序,每一次運(yùn)行的時(shí)候,所分配的進(jìn)程標(biāo)識(shí)符都不相同。
fork
在2.4.4版內(nèi)核中,fork是第2號(hào)系統(tǒng)調(diào)用,其在Linux函數(shù)庫(kù)中的原型是:
#include /* 提供類型pid_t的定義 */
#include /* 提供函數(shù)的定義 */
pid_t fork(void);
只看fork的名字,可能難得有幾個(gè)人可以猜到它是做什么用的。fork系統(tǒng)調(diào)用的作用是復(fù)制一個(gè)進(jìn)程。當(dāng)一個(gè)進(jìn)程調(diào)用它,完成后就出現(xiàn)兩個(gè)幾乎一模一樣的進(jìn)程,我們也由此得到了一個(gè)新進(jìn)程。據(jù)說fork的名字就是來源于這個(gè)與叉子的形狀頗有幾分相似的工作流程。
在Linux中,創(chuàng)造新進(jìn)程的方法只有一個(gè),就是我們正在介紹的fork。其他一些庫(kù)函數(shù),如system(),看起來似乎它們也能創(chuàng)建新的進(jìn)程,如果能看一下它們的源碼就會(huì)明白,它們實(shí)際上也在內(nèi)部調(diào)用了fork。包括我們?cè)诿钚邢逻\(yùn)行應(yīng)用程序,新的進(jìn)程也是由shell調(diào)用fork制造出來的。fork有一些很有意思的特征,下面就讓我們通過一個(gè)小程序來對(duì)它有更多的了解。
/* fork_test.c */
#include
#inlcude
main()
{
pid_t pid;
/*此時(shí)僅有一個(gè)進(jìn)程*/
pid=fork();
/*此時(shí)已經(jīng)有兩個(gè)進(jìn)程在同時(shí)運(yùn)行*/
if(pid
void exit(int status);
不像fork那么難理解,從exit的名字就能看出,這個(gè)系統(tǒng)調(diào)用是用來終止一個(gè)進(jìn)程的。無論在程序中的什么位置,只要執(zhí)行到exit系統(tǒng)調(diào)用,進(jìn)程就會(huì)停止剩下的所有操作,清除包括PCB在內(nèi)的各種數(shù)據(jù)結(jié)構(gòu),并終止本進(jìn)程的運(yùn)行。請(qǐng)看下面的程序:
/* exit_test1.c */
#include
main()
{
printf(“this process will exit!
“);
exit(0);
printf(“never be displayed!
“);
}
編譯后運(yùn)行:
$gcc exit_test1.c -o exit_test1
$./exit_test1
this process will exit!
我們可以看到,程序并沒有打印后面的”never be displayed! “,因?yàn)樵诖酥?,在?zhí)行到exit(0)時(shí),進(jìn)程就已經(jīng)終止了。
exit系統(tǒng)調(diào)用帶有一個(gè)整數(shù)類型的參數(shù)status,我們可以利用這個(gè)參數(shù)傳遞進(jìn)程結(jié)束時(shí)的狀態(tài),比如說,該進(jìn)程是正常結(jié)束的,還是出現(xiàn)某種意外而結(jié)束的,一般來說,0表示沒有意外的正常結(jié)束;其他的數(shù)值表示出現(xiàn)了錯(cuò)誤,進(jìn)程非正常結(jié)束。我們?cè)趯?shí)際編程時(shí),可以用wait系統(tǒng)調(diào)用接收子進(jìn)程的返回值,從而針對(duì)不同的情況進(jìn)行不同的處理。關(guān)于wait的詳細(xì)情況,我們將在以后的篇幅中進(jìn)行介紹。
exit和_exit
作為系統(tǒng)調(diào)用而言,_exit和exit是一對(duì)孿生兄弟,它們究竟相似到什么程度,我們可以從Linux的源碼中找到答案:
#define __NR__exit __NR_exit /* 摘自文件include/a-i386/unistd.h第334行 */
“__NR_”是在Linux的源碼中為每個(gè)系統(tǒng)調(diào)用加上的前綴,請(qǐng)注意之一個(gè)exit前有2條下劃線,第二個(gè)exit前只有1條下劃線。
這時(shí)隨便一個(gè)懂得C語(yǔ)言并且頭腦清醒的人都會(huì)說,_exit和exit沒有任何區(qū)別,但我們還要講一下這兩者之間的區(qū)別,這種區(qū)別主要體現(xiàn)在它們?cè)诤瘮?shù)庫(kù)中的定義。_exit在Linux函數(shù)庫(kù)中的原型是:
#include
void _exit(int status);
和exit比較一下,exit()函數(shù)定義在stdlib.h中,而_exit()定義在unistd.h中,從名字上看,stdlib.h似乎比unistd.h高級(jí)一點(diǎn),那么,它們之間到底有什么區(qū)別呢?讓我們先來看流程圖,通過下圖,我們會(huì)對(duì)這兩個(gè)系統(tǒng)調(diào)用的執(zhí)行過程產(chǎn)生一個(gè)較為直觀的認(rèn)識(shí)。
從圖中可以看出,_exit()函數(shù)的作用最為簡(jiǎn)單:直接使進(jìn)程停止運(yùn)行,清除其使用的內(nèi)存空間,并銷毀其在內(nèi)核中的各種數(shù)據(jù)結(jié)構(gòu);exit()函數(shù)則在這些基礎(chǔ)上作了一些包裝,在執(zhí)行退出之前加了若干道工序,也是因?yàn)檫@個(gè)原因,有些人認(rèn)為exit已經(jīng)不能算是純粹的系統(tǒng)調(diào)用。
exit()函數(shù)與_exit()函數(shù)更大的區(qū)別就在于exit()函數(shù)在調(diào)用exit系統(tǒng)調(diào)用之前要檢查文件的打開情況,把文件緩沖區(qū)中的內(nèi)容寫回文件,就是圖中的”清理I/O緩沖”一項(xiàng)。
在Linux的標(biāo)準(zhǔn)函數(shù)庫(kù)中,有一套稱作”高級(jí)I/O”的函數(shù),我們熟知的printf()、fopen()、fread()、fwrite()都在此列,它們也被稱作”緩沖I/O(buffered I/O)”,其特征是對(duì)應(yīng)每一個(gè)打開的文件,在內(nèi)存中都有一片緩沖區(qū),每次讀文件時(shí),會(huì)多讀出若干條記錄,這樣下次讀文件時(shí)就可以直接從內(nèi)存的緩沖區(qū)中讀取,每次寫文件的時(shí)候,也僅僅是寫入內(nèi)存中的緩沖區(qū),等滿足了一定的條件(達(dá)到一定數(shù)量,或遇到特定字符,如換行符和文件結(jié)束符EOF),再將緩沖區(qū)中的內(nèi)容一次性寫入文件,這樣就大大增加了文件讀寫的速度,但也為我們編程帶來了一點(diǎn)點(diǎn)麻煩。如果有一些數(shù)據(jù),我們認(rèn)為已經(jīng)寫入了文件,實(shí)際上因?yàn)闆]有滿足特定的條件,它們還只是保存在緩沖區(qū)內(nèi),這時(shí)我們用_exit()函數(shù)直接將進(jìn)程關(guān)閉,緩沖區(qū)中的數(shù)據(jù)就會(huì)丟失,反之,如果想保證數(shù)據(jù)的完整性,就一定要使用exit()函數(shù)。
請(qǐng)看以下例程:
/* exit2.c */
#include
main()
{
printf(“output begin
“);
printf(“content in buffer”);
exit(0);
}
編譯并運(yùn)行:
$gcc exit2.c -o exit2
$./exit2
output begin
content in buffer
/* _exit1.c */
#include
main()
{
printf(“output begin
“);
printf(“content in buffer”);
_exit(0);
}
編譯并運(yùn)行:
$gcc _exit1.c -o _exit1
$./_exit1
output begin
在Linux中,標(biāo)準(zhǔn)輸入和標(biāo)準(zhǔn)輸出都是作為文件處理的,雖然是一類特殊的文件,但從程序員的角度來看,它們和硬盤上存儲(chǔ)數(shù)據(jù)的普通文件并沒有任何區(qū)別。與所有其他文件一樣,它們?cè)诖蜷_后也有自己的緩沖區(qū)。
請(qǐng)讀者結(jié)合前面的敘述,思考一下為什么這兩個(gè)程序會(huì)得出不同的結(jié)果。相信如果您理解了我前面所講的內(nèi)容,會(huì)很容易的得出結(jié)論。
在這篇文章中,我們對(duì)Linux的進(jìn)程管理作了初步的了解,并在此基礎(chǔ)上學(xué)習(xí)了getpid、fork、exit和_exit四個(gè)系統(tǒng)調(diào)用。在下一篇文章中,我們將學(xué)習(xí)與Linux進(jìn)程管理相關(guān)的其他系統(tǒng)調(diào)用,并將作一些更深入的探討。
前面的文章中,我們已經(jīng)了解了父進(jìn)程和子進(jìn)程的概念,并已經(jīng)掌握了系統(tǒng)調(diào)用exit的用法,但可能很少有人意識(shí)到,在一個(gè)進(jìn)程調(diào)用了exit之后,該進(jìn)程并非馬上就消失掉,而是留下一個(gè)稱為僵尸進(jìn)程(Zombie)的數(shù)據(jù)結(jié)構(gòu)。在Linux進(jìn)程的5種狀態(tài)中,僵尸進(jìn)程是非常特殊的一種,它已經(jīng)放棄了幾乎所有內(nèi)存空間,沒有任何可執(zhí)行代碼,也不能被調(diào)度,僅僅在進(jìn)程列表中保留一個(gè)位置,記載該進(jìn)程的退出狀態(tài)等信息供其他進(jìn)程收集,除此之外,僵尸進(jìn)程不再占有任何內(nèi)存空間。從這點(diǎn)來看,僵尸進(jìn)程雖然有一個(gè)很酷的名字,但它的影響力遠(yuǎn)遠(yuǎn)抵不上那些真正的僵尸兄弟,真正的僵尸總能令人感到恐怖,而僵尸進(jìn)程卻除了留下一些供人憑吊的信息,對(duì)系統(tǒng)毫無作用。
也許讀者們還對(duì)這個(gè)新概念比較好奇,那就讓我們來看一眼Linux里的僵尸進(jìn)程究竟長(zhǎng)什么樣子。
當(dāng)一個(gè)進(jìn)程已退出,但其父進(jìn)程還沒有調(diào)用系統(tǒng)調(diào)用wait(稍后介紹)對(duì)其進(jìn)行收集之前的這段時(shí)間里,它會(huì)一直保持僵尸狀態(tài),利用這個(gè)特點(diǎn),我們來寫一個(gè)簡(jiǎn)單的小程序:
/* zombie.c */
#include
#include
main()
{
pid_t pid;
pid=fork();
if(pid
printf(“error occurred!n”);
else if(pid==0) /* 如果是子進(jìn)程 */
exit(0);
else /* 如果是父進(jìn)程 */
sleep(60); /* 休眠60秒,這段時(shí)間里,父進(jìn)程什么也干不了 */
wait(NULL); /* 收集僵尸進(jìn)程 */
}
sleep的作用是讓進(jìn)程休眠指定的秒數(shù),在這60秒內(nèi),子進(jìn)程已經(jīng)退出,而父進(jìn)程正忙著睡覺,不可能對(duì)它進(jìn)行收集,這樣,我們就能保持子進(jìn)程60秒的僵尸狀態(tài)。
編譯這個(gè)程序:
$ cc zombie.c -o zombie
后臺(tái)運(yùn)行程序,以使我們能夠執(zhí)行下一條命令:
$ ./zombie &
1577
列一下系統(tǒng)內(nèi)的進(jìn)程:
$ ps -ax
… …
1177 pts/0 S:00 -bash
1577 pts/0 S:00 ./zombie
1578 pts/0 Z:00
1579 pts/0 R:00 ps -ax
看到中間的”Z”了嗎?那就是僵尸進(jìn)程的標(biāo)志,它表示1578號(hào)進(jìn)程現(xiàn)在就是一個(gè)僵尸進(jìn)程。
我們已經(jīng)學(xué)習(xí)了系統(tǒng)調(diào)用exit,它的作用是使進(jìn)程退出,但也僅僅限于將一個(gè)正常的進(jìn)程變成一個(gè)僵尸進(jìn)程,并不能將其完全銷毀。僵尸進(jìn)程雖然對(duì)其他進(jìn)程幾乎沒有什么影響,不占用CPU時(shí)間,消耗的內(nèi)存也幾乎可以忽略不計(jì),但有它在那里呆著,還是讓人覺得心里很不舒服。而且Linux系統(tǒng)中進(jìn)程數(shù)目是有限制的,在一些特殊的情況下,如果存在太多的僵尸進(jìn)程,也會(huì)影響到新進(jìn)程的產(chǎn)生。那么,我們?cè)撊绾蝸硐麥邕@些僵尸進(jìn)程呢?
先來了解一下僵尸進(jìn)程的來由,我們知道,Linux和UNIX總有著剪不斷理還亂的親緣關(guān)系,僵尸進(jìn)程的概念也是從UNIX上繼承來的,而UNIX的先驅(qū)們?cè)O(shè)計(jì)這個(gè)東西并非是因?yàn)殚e來無聊想煩煩其他的程序員。僵尸進(jìn)程中保存著很多對(duì)程序員和系統(tǒng)管理員非常重要的信息,首先,這個(gè)進(jìn)程是怎么死亡的?是正常退出呢,還是出現(xiàn)了錯(cuò)誤,還是被其它進(jìn)程強(qiáng)迫退出的?其次,這個(gè)進(jìn)程占用的總系統(tǒng)CPU時(shí)間和總用戶CPU時(shí)間分別是多少?發(fā)生頁(yè)錯(cuò)誤的數(shù)目和收到信號(hào)的數(shù)目。這些信息都被存儲(chǔ)在僵尸進(jìn)程中,試想如果沒有僵尸進(jìn)程,進(jìn)程一退出,所有與之相關(guān)的信息都立刻歸于無形,而此時(shí)程序員或系統(tǒng)管理員需要用到,就只好干瞪眼了。
linux中C語(yǔ)言關(guān)于進(jìn)程的創(chuàng)建
父進(jìn)程調(diào)用完子進(jìn)程后,調(diào)用wait阻塞自己 ,等待子進(jìn)程執(zhí)行完畢再執(zhí)行之后的操作。
/*請(qǐng)解釋一下*/
while(((child=wait(&status))==-1)&(errno==EINTR));
這種的目的是父親進(jìn)程等待子進(jìn)程結(jié)枯困束,并回收子進(jìn)程的資源,將子進(jìn)程的退出狀碼桐態(tài)存儲(chǔ)在status中,同時(shí),返回該子進(jìn)程的pid。
如果wait函數(shù)返回-1表示wait函數(shù)被其它情況打斷返回,并沒有等待到子進(jìn)程結(jié)束,而同時(shí)判斷errno的值是不是EINTR(意思是讓你try again),那么,讓進(jìn)程繼續(xù)等待。因?yàn)檫@個(gè)錯(cuò)誤并不是真正wait錯(cuò)誤,而是被timeout時(shí)間等造成的,因此重新等待。而如果是其它情況,顯然是wait函數(shù)調(diào)用錯(cuò)誤,即遲敗坦下面的if(child==-1),需要打印錯(cuò)誤信息?!?/p>
//但你這句應(yīng)該寫錯(cuò)了。應(yīng)該是邏輯與而不是位與操作。即
while(((child=wait(&status))==-1)&&(errno==EINTR));
/*請(qǐng)解釋一下*/
if(child==-1)
你對(duì)信號(hào)處理部分還需要努力。
另外介紹一本書《Linux高級(jí)程序設(shè)計(jì) 第3版》 上面講得很清楚。
有問題我們繼續(xù)交流,一起學(xué)習(xí)。
更多技術(shù)文章可以關(guān)注我的微博,名字:成都睿爾科技 。
1、fork 是用來創(chuàng)建子進(jìn)程的, 而不是線程( 線程創(chuàng)頃世鏈建需要用到 pthread_create )。
需要根據(jù)返滲 fork() 的返回值來判斷下面的代碼是在父進(jìn)程(返回pid>0)中還是子進(jìn)程(返回0)中. 像上面的代碼中 if 中的代碼被在子進(jìn)程中執(zhí)行, else 中的代碼在父進(jìn)程中執(zhí)行。
2、例程:
#include
#include
#include
int main()
{
pid_t id; //定義一個(gè)進(jìn)程號(hào)變量
int i=0;
printf(“start fork/n”);
id = fork(); //調(diào)用fork函數(shù)新建一個(gè)進(jìn)程
i ++;
printf(“end fork/n”);
//判斷當(dāng)前進(jìn)程
if(id
perror(“fork failed/n”);
exit(1);
}
else if(id == 0){ //子進(jìn)程
printf(“In child/n”);
printf(“i = %d/n”, i++);
exit(0);
}
else{ //父進(jìn)程
printf(“In father/n”);
printf(“i = %d/n”雀孫, i++);
exit(0);
}
return 0;
}
這根信號(hào)有關(guān)系。父進(jìn)程調(diào)用wait會(huì)阻塞 自己,直到子進(jìn)程退出。
例外是父進(jìn)程阻罩檔差塞過程中收到信號(hào),比如收到一個(gè)CTRL-C組合鍵, 或是收到一個(gè)Kill信號(hào)等,這時(shí)父進(jìn)程退出阻塞,去執(zhí)行自己的蠢團(tuán)信號(hào)處理程序。執(zhí)行之后如果程序沒有退出,就會(huì)到達(dá)這里的while語(yǔ)句。wait返回-1,物皮 errno=EINTR. 因?yàn)榈却臈l件沒發(fā)生, 所以遇到這個(gè)條件, 程序重新執(zhí)行wait.
linux父進(jìn)程pid代碼的介紹就聊到這里吧,感謝你花時(shí)間閱讀本站內(nèi)容,更多關(guān)于linux父進(jìn)程pid代碼,深入理解Linux:父進(jìn)程PID代碼解析,linux下的PID,PIDD是什么?他們之間的關(guān)系以及應(yīng)用是什么?,linux中C語(yǔ)言關(guān)于進(jìn)程的創(chuàng)建的信息別忘了在本站進(jìn)行查找喔。
成都網(wǎng)站設(shè)計(jì)制作選創(chuàng)新互聯(lián),專業(yè)網(wǎng)站建設(shè)公司。
成都創(chuàng)新互聯(lián)10余年專注成都高端網(wǎng)站建設(shè)定制開發(fā)服務(wù),為客戶提供專業(yè)的成都網(wǎng)站制作,成都網(wǎng)頁(yè)設(shè)計(jì),成都網(wǎng)站設(shè)計(jì)服務(wù);成都創(chuàng)新互聯(lián)服務(wù)內(nèi)容包含成都網(wǎng)站建設(shè),小程序開發(fā),營(yíng)銷網(wǎng)站建設(shè),網(wǎng)站改版,服務(wù)器托管租用等互聯(lián)網(wǎng)服務(wù)。
當(dāng)前題目:深入理解Linux:父進(jìn)程PID代碼解析 (linux父進(jìn)程pid代碼)
URL地址:http://fisionsoft.com.cn/article/dhjpgji.html


咨詢
建站咨詢
