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