先回忆一下 fork() 函数
由fork创建的新进程被称为子进程(child process)。fork函数被调用一次,但返回两次。
1.在父进程中,fork返回新创建子进程的进程ID。
2.在子进程中,fork返回0。
3.如果出现错误,fork返回一个负值。
而且
在fork()调用处父进程空间会复制到子进程中,包括:指令、调用值、栈、环境变量、缓冲区等。(这句不一定正确)
#include<stdio.h>
#include<stdlib.h>
#include<sys/types.h>
#include<unistd.h>
int main(void)
{
int i;
for(i = 0 ; i < 2; i++)
{
fork();
printf("_");
}
return 0;
}
问:这里一共输出了多少个"_"?
答案:8个
printf某些内容时,操作系统仅仅是把该内容放到了stdout的缓冲队列里了,并没有实际的写到屏幕上。但是,只要看到有/n 或是EOF,或是主动刷出,或是缓冲区满,或是文件描述符关闭,或是exit,就会把数据从缓冲区中输出出来(flush)。也就是说对于printf()代表的行缓冲而言,数据只是先被存储在了缓冲区,直到碰见上述情况才会输出。如果缓冲区内有内容,注意,当进程退出时(exit)也会自动进行输出,即使进程中并没有printf()语句。
运行了printf("_")后,下划线 仅仅被放到了缓冲区里,程序运行到fork时缓冲区里面的的下划线也会被子进程复制过去。因此在子进程的stdout缓冲中也就有了之前的下划线。所以最终会输出两个下划线。
上图中
深色的为第一次 i=0 第一次并没有输出,而是在缓冲区(云状图形中)并且在各自创建子进程的时候顺带复制了到子进程中。
浅色的为第二次 i=1 因为 i<2 所以这次是最后一次执行,红框部分即为实际输出的部分。