與fork()函數有所不同,fork()函數在建立一個子進程後,子進程的地址空間完全和父進程分開。父子進程是兩個獨立的進程,接受系統調度和分配的機會均等。因此父進程和子進程更像是一對兄弟。
而vfork()函數不同,vfork()函數產生的子進程和父進程完全共用地址空間,包括程式碼片段,資料區段和堆棧段。子進程對這些共用資源的修改會影響父進程,vfork()產生的更像是一個線程。再者,vfork()函數產生的進程一定比父進程先運行,子進程運行完了以後,父進程再運行。
看段代碼:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int goal=1;
int main()
{
pid_t pid;
int a=2;
int *p;
p=(int *)malloc(sizeof(int));
*p=6;
pid=vfork();
if(pid<0){exit(1);}
else if(pid==0)
{
goal=goal+3;
a=a+7;
(*p)=(*p)+8;
printf("%d,%d,%d\n",goal,a,*p);
exit(1);
}
sleep(2);
printf("%d,%d,%d\n",goal,a,*p);
運行結果:
4,9,14
4,9,14
使用vfork()函數時應當注意不要在子函數內調用它。
看段代碼:
#include<stdio.h>
#include<unistd.h>
int f1()
{vfork();
return 0;}
int f2(int a,int b)
{
return a+b;
}
int main()
{
int c;
f1();
c=f2(1,2);
printf("%d\n",c);
return 0;
}
運行結果:
3
段錯誤(吐核)
錯誤的原因是:兩個進程共用堆棧段,兩個進程都要從f1函數中返回。由於子進程先於父進程運行,所以子進程從f1返回,並且調用f2函數。其棧幀覆蓋了原來的f1函數的棧幀,當子進程結束時父進程運行時,就出錯了。。。由此可知,使用vfork()函數之後,子進程對父進程的影響是巨大的,其同步措施勢在必行,同時也可以體會到,線程的同步是多麼的重要!!!
(文中內容基本出自《Linux c程式設計大全》吳嶽)