fork:
fork()后会有两个并发进程执行,子进程复制了父进程的数据段,包括全局变量。父进程返回子进程的PID, 子进程返回0。 头文件为#include<unistd.h>
父进程和子进程中的内存变量等是相互独立的。(父进程中的变量是父进程的,子进程的变量是子进程的,两者互不干扰。)
父进程、子进程不共享内存,不共享变量。
子进程是父进程的精确复制,两者执行相同的代码,但内存各是各的,不共享数据、变量什么的,内存相互独立。
vfork:
父进程和子进程共享内存。
子进程与父进程共享地址空间。调用vfork()创建的子进程后,父进程被挂起,直到子进程结束。头文件:stdio.h
1,fork
进程由 fork 函数创建,在 unistd.h 库中定义如下
#include<unistd.h>
pid_t fork ( void ) ;
(1)fork调用的一个奇妙之处就是它仅仅被调用一次,却能够返回两次,它可能有三种不同的返回值:
1)在父进程中,fork返回新创建子进程的进程ID;
2)在子进程中,fork返回0;
3)如果出现错误,fork返回一个负值;
1) >0: 表示系统将执行父进程的程序段。
2) =0: 表示系统将执行子进程的有关程序段。
3 )-1: 表示子进程未创建成功。
(2)子进程、父进程不共享内存。
fork 创建一个新进程(子进程),除了子进程标识符和某些特性参数不同之外,子进程是父进程的精确复制。
子进程和父进程都执行在 fork 函数调用之后的代码,子进程是父进程的一个拷贝。例如,父进程的数据空间、堆栈空间都会给子进程一个拷贝,而不是共享这些内存。
2,vfork
vfork系统调用不同于fork,用vfork创建的子进程与父进程共享地址空间,也就是说子进程完全运行在父进程的地址空间上,如果这时子进程修改了某个变量,这将影响到父进程(与fork不同,fork子进程的变量改变不会影响到父进程)。
因此,如果改用vfork()的话,那么两次打印a,b的值是相同的,所在地址也是相同的。
但此处有一点要注意的是用vfork()创建的子进程必须显示调用exit()来结束,否则子进程将不能结束,而fork()则不存在这个情况。
Vfork也是在父进程中返回子进程的进程号,在子进程中返回0。
用 vfork创建子进程后,父进程会被阻塞直到子进程调用exec(exec,将一个新的可执行文件载入到地址空间并执行之。)或exit。vfork的好处是在子进程被创建后往往仅仅是为了调用exec执行另一个程序,因为它就不会对父进程的地址空间有任何引用,所以对地址空间的复制是多余的 ,因此通过vfork共享内存可以减少不必要的开销。
3,clone
系统调用fork()和vfork()是无参数的,而clone()则带有参数。fork()是全部复制,vfork()是共享内存,而clone()是则可以将父进程资源有选择地复制给子进程,而没有复制的数据结构则通过指针的复制让子进程共享,具体要复制哪些资源给子进程,由参数列表中的clone_flags来决定。另外,clone()返回的是子进程的pid。