c语言中fork,C语言fork函数学习
Fork概念
一個進(jìn)程,包括代碼、數(shù)據(jù)和分配給進(jìn)程的資源。fork()函數(shù)通過系統(tǒng)調(diào)用創(chuàng)建一個與原來進(jìn)程幾乎完全相同的進(jìn)程,也就是兩個進(jìn)程可以做完全相同的事,但如果初始參數(shù)或者傳入的變量不同,兩個進(jìn)程也可以做不同的事。
一個進(jìn)程調(diào)用fork()函數(shù)后,系統(tǒng)先給新的進(jìn)程分配資源,例如存儲數(shù)據(jù)和代碼的空間。然后把原來的進(jìn)程的所有值都復(fù)制到新的新進(jìn)程中,只有少數(shù)值與原來的進(jìn)程的值不同。相當(dāng)于克隆了一個自己。
由fork函數(shù)創(chuàng)建的新進(jìn)程被稱為子進(jìn)程。fork函數(shù)被調(diào)用一次,但是返回兩次。父進(jìn)程返回的值是新進(jìn)程的進(jìn)程ID,而子進(jìn)程返回的值是0。
fork函數(shù)返回值的三種情況
返回子進(jìn)程Id給父進(jìn)程
因?yàn)橐粋€進(jìn)程的子進(jìn)程可能有多個,并且沒有一個函數(shù)可以獲得一個進(jìn)程的所有子進(jìn)程ID。
返回給子進(jìn)程值為0
一個進(jìn)程只會有一個父進(jìn)程,所以子進(jìn)程總是可以調(diào)用getpid以獲得當(dāng)前進(jìn)程Id以及調(diào)用getppid獲得父進(jìn)程Id.
出現(xiàn)錯誤,返回負(fù)值
當(dāng)前進(jìn)程數(shù)已經(jīng)達(dá)到系統(tǒng)規(guī)定的上限,這時errno的值被設(shè)置為EAGAIN
系統(tǒng)內(nèi)存不足,這時errno的值被設(shè)置為ENOMEM
創(chuàng)建新進(jìn)程成功后,系統(tǒng)中出現(xiàn)兩個基本完全相同的進(jìn)程,這兩個進(jìn)程執(zhí)行沒有固定的先后順序,哪個進(jìn)程先執(zhí)行要看系統(tǒng)的進(jìn)程調(diào)度策略
子進(jìn)程執(zhí)行代碼開始位置
fork確實(shí)創(chuàng)建可一個子進(jìn)程并完全復(fù)制父進(jìn)程,但是子進(jìn)程是從fork后面到那個指令開始執(zhí)行。如果子進(jìn)程也從main開頭到尾執(zhí)行所有指令,那么它執(zhí)行到fork指令時也必定會創(chuàng)建一個個子子進(jìn)程,子子孫孫無窮盡。
常見的兩種應(yīng)用場景
一個父進(jìn)程希望復(fù)制自己,使父、子進(jìn)程同時執(zhí)行不同的代碼段。這在網(wǎng)絡(luò)服務(wù)中是常見的。
-父進(jìn)程等待客戶端的服務(wù)請求,當(dāng)這種請求到達(dá)時,父進(jìn)程調(diào)用fork,使子進(jìn)程處理此請求。父進(jìn)程則繼續(xù)等待下一個服務(wù)請求的到達(dá)
一個進(jìn)程要執(zhí)行一個不同的程序。這是shell中常見的情況,子進(jìn)程從fork返回后立即調(diào)用exec
示例
示例1:
#include
#include
#include
#include
int main(int argc, char *argv[]) {
pid_t pid;
int count = 0;
//獲得當(dāng)前進(jìn)程ID
printf("Current Process Id = %d \n", getpid());
if ((pid = fork()) < 0) {
printf("異常退出");
exit(1);
} else if (pid == 0) {
count++;
printf("進(jìn)入子進(jìn)程, 當(dāng)前進(jìn)程 currentPid = %d, 父進(jìn)程 parentPid = %d \n", getpid(),getppid());
} else {
count++;
printf("當(dāng)前進(jìn)程 當(dāng)前進(jìn)程 currentPid = %d, 子進(jìn)程 childPid = %d \n", getpid(), pid);
}
printf("當(dāng)前進(jìn)程 currentPid = %d, Count = %d \n", getpid(), count);
return 0;
}
fork創(chuàng)建了子進(jìn)程,并復(fù)制一個count,從最后的可以看出各個子進(jìn)程之間的count不相關(guān)。
示例2:
#include
#include
#include
#include
int main(int argc, char *argv[]) {
pid_t pid;
int loop;
//獲得當(dāng)前進(jìn)程ID
printf("Current Process Id = %d \n", getpid());
for (loop = 0; loop < 2; loop++) {
//fork創(chuàng)建子進(jìn)程
if ((pid = fork()) < 0) {
printf("退出程序\n");
exit(1);
} else if (pid == 0) {
//創(chuàng)建子進(jìn)程后,子進(jìn)程返回0,通過getPid得到當(dāng)前進(jìn)程,getPPid得到父進(jìn)程
printf("Child Process loop = %d, Current Pid = %d , Parent Pid = %d\n", loop, getpid(), getppid());
} else {
//創(chuàng)建子進(jìn)程后,pid就是子進(jìn)程ID
printf("Child Process loop = %d, Pid = %d,Parent Pid = %d\n", loop, pid, getpid());
sleep(5);
}
}
return 0;
}
編譯運(yùn)行后的結(jié)果:
創(chuàng)建了三個子進(jìn)程:
參考
總結(jié)
以上是生活随笔為你收集整理的c语言中fork,C语言fork函数学习的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 数字图像处理使用计算机对,数字图像处理的
- 下一篇: Matlab的最优化实现方法之线性规划: