最近在看進(jìn)程間的通信,看到了fork()函數(shù),雖然以前用過(guò),這次經(jīng)過(guò)思考加深了理解,F(xiàn)總結(jié)如下:
1.函數(shù)本身
(1)頭文件
#include<unistd.h>
#include<sys/types.h>
(2)函數(shù)原型
pid_t fork( void);
。╬id_t 是一個(gè)宏定義,其實(shí)質(zhì)是int 被定義在#include<sys/types.h>中)
返回值: 若成功調(diào)用一次則返回兩個(gè)值,子進(jìn)程返回0,父進(jìn)程返回子進(jìn)程ID;否則,出錯(cuò)返回-1
(3)函數(shù)說(shuō)明
一個(gè)現(xiàn)有進(jìn)程可以調(diào)用fork函數(shù)創(chuàng)建一個(gè)新進(jìn)程。由fork創(chuàng)建的新進(jìn)程被稱(chēng)為子進(jìn)程(child process)。子進(jìn)程是父進(jìn)程的副本,它將獲得父進(jìn)程數(shù)據(jù)空間、堆、棧等資源的副本。注意,子進(jìn)程持有的是上述存儲(chǔ)空間的“副本”,這意味著父子進(jìn)程間不共享這些存儲(chǔ)空間,子進(jìn)程有了獨(dú)立的地址空間。
2.代碼執(zhí)行解釋
(1)代碼如下圖所示
(2)分析
由操作系統(tǒng)相關(guān)知識(shí)可知,進(jìn)程是系統(tǒng)資源分配的基本單位,因此子進(jìn)程與父進(jìn)程不共享進(jìn)程資源空間。在執(zhí)行代碼段第8行之前,系統(tǒng)中只有默認(rèn)的主進(jìn)程。在執(zhí)行完代碼段第8行后,系統(tǒng)中就有了兩個(gè)進(jìn)程,即主進(jìn)程和由其創(chuàng)建的子進(jìn)程。
創(chuàng)建子進(jìn)程,fork()函數(shù)返回兩個(gè)數(shù)值,若創(chuàng)建成功,子進(jìn)程中返回0;父進(jìn)程返回子進(jìn)程ID。用資源空間圖示如下:
執(zhí)行了fork()函數(shù)后,主進(jìn)程為父進(jìn)程生成了一份資源空間的副本。主進(jìn)程中的pid為子進(jìn)程的pid(pid>0),子進(jìn)程中的pid為0。
在fork()函數(shù)之后父進(jìn)程與子進(jìn)程都從下一行執(zhí)行,即第9行。因?yàn)橹鬟M(jìn)程中pid>0,可以執(zhí)行else if(pid>0)段代碼,子進(jìn)程pid=0,可以執(zhí)行else if(pid==0)段代碼。
(3)代碼執(zhí)行結(jié)果如下:
可見(jiàn),"Before the fork ..."只執(zhí)行了一次。"After the fork ..."執(zhí)行了兩次。
。ň唧w的執(zhí)行結(jié)果,可能會(huì)由于進(jìn)程調(diào)度的不同,后面的四個(gè)輸出順序可能不同。不過(guò)第一個(gè)輸出的一定是"Before the fork ...")。