fork、vfork、clone 三者的区别

fork:

         fork()后会有两个并发进程执行,子进程复制了父进程的数据段,包括全局变量。父进程返回子进程的PID, 子进程返回0。 头文件为stdio.h

        父进程和子进程中的内存变量等是相互独立的。(父进程中的变量是父进程的,子进程的变量是子进程的,两者互不干扰。)

         父进程、子进程不共享内存,不共享变量。

        子进程是父进程的精确复制,两者执行相同的代码,但内存各是各的,不共享数据、变量什么的,内存相互独立。

vfork:

          父进程和子进程共享内存

          子进程与父进程共享地址空间。调用vfork()创建的子进程后,父进程被挂起,直到子进程结束。头文件:stdio.h

1,fork

进程由 fork 函数创建,在 unistd.h 库中定义如下
# i n c l u d e < u n i st d . h>
p i d _ t f o r k ( 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。
 

猜你喜欢

转载自blog.csdn.net/weixin_39450145/article/details/83719913