记得之前初次接触fork()函数的时候,一直被“printf”输出多少次的问题弄得比较晕乎。不过,“黄天不负留心人"。哈~ 终于在学习进程和进程建立fork相关知识后,总算是大体摸清了其中的前因后果。废话很少讲,下面来谈谈本人的一点小小积累linux
1 /************************************************************************* 2 > File Name: 1.c 3 > Author: tp 4 > Mail: 5 > Created Time: Mon 07 May 2018 07:57:28 PM CST 6 ************************************************************************/ 7 8 #include <stdio.h> 9 #include <stdlib.h> 10 #include <unistd.h> 11 int main( void) 12 { 13 printf("change world!\n"); 14 pid_t pid = fork(); 15 if( pid == -1) {perror("fork"),exit(1); } 16 17 printf( "pid=%d, returnVal=%d\n", getpid(), pid); 18 sleep( 1); 19 exit(0); 20 } ~
1 /************************************************************************* 2 > File Name: 2.c 3 > Author: tp 4 > Mail: 5 > Created Time: Mon 07 May 2018 12:40:39 PM CST 6 ************************************************************************/ 7 8 #include <stdio.h> 9 #include <stdlib.h> 10 #include <unistd.h> 11 #include <fcntl.h> 12 13 int set = 110; 14 int main( void) 15 { 16 printf( "before fork\n"); 17 pid_t pid = fork( ); 18 if( pid < 0){ perror(" fork"),exit( 1);} 19 20 if( pid == 0) 21 { 22 ++set; 23 printf( "son pid=%d, %d\n", getpid(), set); 24 } 25 else 26 { 27 sleep( 1); 28 printf( "parent pid=%d , %d\n", getpid( ), set); 29 } 30 exit( 0); 31 }
看一下结果:算法
不难注意到 “before fork”这句话只是被打印了一次,这个从上面的例子,这不难理解;与此同时子进程中的set的值被改变了。此时再进行一个重定向操做会发生什么网络

出现很神奇的现象! 这个时候打印了出了两次“before fork”,不只仅是如此,上述针对父进程的标准输出执行重定向操做还致使了子进程也执行重定向操做。数据结构
透过现象看本质,来细细分析一下。针对打印两次“before fork”,首先,先要知道标准IO库是是带缓冲的,而像printf这种直接输出到标准输出时,这个缓冲区是由换行符刷新的;而当执行了重定向操做,这里就是将标准输出重定向到文件,文件就不会当即去刷新缓冲区(全缓冲的方式);好,因为在fork以前调用了一次printf,但fork以后,该行数据仍存留在缓冲区中,而后父进程数据空间被复制到子进程中,该行数据去也被复制了过去,这样父子进程都各自带有该行内容的缓冲区了,至关于子进程缓冲区添加了一行“before fork”,而后在每一个进程exit以后,每一个缓冲区的内容就被写到了相应的文件中。函数
再一个就是,在重定向父进程的标准输出时,子进程标准输出也被重定向。这就源于父子进程会共享全部的打开文件。 由于fork的特性就是将父进程全部打开文件描述符复制到子进程中。当父进程的标准输出被重定向,子进程本是写到标准输出的时候,此时天然也改写到那个对应的地方;与此同时,在父进程等待子进程执行时,子进程被改写到文件show.out中,而后又更新了与父进程共享的该文件的偏移量;那么在子进程终止后,父进程也写到show.out中,同时其输出还会追加在子进程所写数据以后,这也就解释了上面为何“before fork”会在一个文件中打印两次。性能
在fork以后处理文件描述符通常又如下两种状况:学习
1.父进程等待子进程完成。此种状况,父进程无需对其描述符做任何处理。当子进程终止后,它曾进行过读,写操做的任一共享描述符的文件偏移已发生改变。spa
2.父子进程各自执行不一样的程序段。这样fork以后,父进程和子进程各自关闭它们再也不使用的文件描述符,这样就避免干扰对方使用的文件描述符了。这相似于网络服务进程。.net
同时父子进程也是有区别的:它们不只仅是两个返回值不一样;它们各自的父进程也不一样,父进程的父进程是ID不变的;还有子进程不继承父进程设置的文件锁,子进程未处理的信号集会设置为空集等不一样code
vfork和fork之间的区别: