創(chuàng)建一個新進(jìn)程的方法只有由某個已存在的進(jìn)程調(diào)用fork()或vfork(),當(dāng)然某些進(jìn)程如init等是作為系統(tǒng)啟動的一部風(fēng)而被內(nèi)核創(chuàng)建的。
1.fork函數(shù)介紹
#include <sys/types.h>
#include <unistd.h>
pid_t fork (void );
正確返回:父進(jìn)程中返回子進(jìn)程的進(jìn)程號;子進(jìn)程中返回0;(單調(diào)用雙返回函數(shù))
錯誤返回:-1;
子進(jìn)程是父進(jìn)程的一個拷貝。具體說,子進(jìn)程從父進(jìn)程那得到了數(shù)據(jù)段和堆棧段,但不是與父進(jìn)程共享而是單獨(dú)分配內(nèi)存。fork函數(shù)返回后,子進(jìn)程和父進(jìn)程都是從fork函數(shù)的下一條語句開始執(zhí)行。
由于子進(jìn)程與父進(jìn)程的運(yùn)行是無關(guān)的,父進(jìn)程可先于子進(jìn)程運(yùn)行,子進(jìn)程也可先于父進(jìn)程運(yùn)行,所以下段程序可以有兩種運(yùn)行結(jié)果。
[root@happy src]# cat simplefork.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int globa = 4;
int main (void )
{
pid_t pid;
int vari = 5;
printf ("before fork\n" );
if ((pid = fork()) < 0){
printf ("fork error\n");
exit (0);
}
else if (pid == 0){
globa++ ;
vari--;
printf("Child changed\n");
}
else
printf("Parent did not changde\n");
printf("globa = %d vari = %d\n",globa,vari);
exit(0);
}
運(yùn)行結(jié)果:(可能出現(xiàn)的一種結(jié)果)
[root@happy src]# ./a.out
before fork
Child changed
globa = 5 vari = 4
Parent did not changde
globa = 4 vari = 5
2.vfork函數(shù)介紹
vfork創(chuàng)建新進(jìn)程的主要目的在于用exec函數(shù)執(zhí)行另外的程序,實(shí)際上,在沒調(diào)用exec或exit之前子進(jìn)程的運(yùn)行中是與父進(jìn)程共享數(shù)據(jù)段的。在vfork調(diào)用中,子進(jìn)程先運(yùn)行,父進(jìn)程掛起,直到子進(jìn)程調(diào)用exec或exit,在這以后,父子進(jìn)程的執(zhí)行順序不再有限制。
[root@happy src]# cat simplevfork.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int globa = 4;
int main (void )
{
pid_t pid;
int vari = 5;
if ((pid = vfork()) < 0){
printf ("fork error\n");
exit (0);
}
else if (pid == 0){
globa++ ;
vari--;
printf("Child changed\n");
_exit(0);
}
else
printf("Parent did not changde\n");
printf("globa = %d vari = %d\n",globa,vari);
exit(0);
}
運(yùn)行結(jié)果(固定的):
[root@happy src]# gcc simplevfork.c
[root@happy src]# ./a.out
Child changed
Parent did not changde
globa = 5 vari = 4
總結(jié)
區(qū)別:
1、fork()用于創(chuàng)建一個新進(jìn)程。由fork()創(chuàng)建的子進(jìn)程是父進(jìn)程的副本。即子進(jìn)程獲取父進(jìn)程數(shù)據(jù)空間,堆和 棧的副本。父子進(jìn)程之間不共享這些存儲空間的部分。而vfork()創(chuàng)建的進(jìn)程并不將父進(jìn)程的地址空間完全復(fù)制到子進(jìn)程中,因?yàn)樽舆M(jìn)程會立即調(diào)用exec (或exit)于是也就不會存放該地址空間。相反,在子進(jìn)程調(diào)用exec或exit之前,它在父進(jìn)程的空間進(jìn)行。
2、vfork()與fork()另一個區(qū)別就是:vfork保證子進(jìn)程先運(yùn)行,在它調(diào)用exec或exit之后父進(jìn)程才可能被調(diào)度運(yùn)行。
相同:
1、兩者被調(diào)用一次,但是返回兩次。兩次返回的唯一區(qū)別是子進(jìn)程的返回值是0,而父進(jìn)程的返回值則是新子進(jìn)程的進(jìn)程ID。