一直对于循环语句中调用fork创建子进程的过程不甚了解,直到在笔试中遇到了一道题目,因此简单做下笔记。
原题:
#include<stdio.h>
#include<unistd.h>
int main(int argc,char **argv)
{
for(int i=0;i<2;++i)
{
fork();
printf("*");
}
return 0;
}
请问输出了多少个*
。
当时写的是6个,但事后测试发现是8个。
因此在CSDN上找大佬寻求解答。
解释:fork()系统调用是Unix下以自身进程创建子进程的系统调用,一次调用,两次返回,如果返回是0,则是子进程,如果返回值>0,则是父进程(返回值是子进程的pid)。在fork()的调用处,整个父进程空间会原模原样地复制到子进程中,包括指令,变量值,程序调用栈,环境变量,缓冲区,等等。先看改进代码,观察进程树:
有8个’-’是因为printf(“-”);语句有buffer;在fork的时候,缓存被复制到了子进程空间,所以,就多了两个。下图阴影双边框的两个子进程复制了父进程缓冲区的‘-‘
- 当i=0时,fork()创建了一个子进程,printf函数还没有执行,因此缓冲区中没有数据,父子进程打印了
-
- 当i=1时,父进程又fork了一个子进程,但是此时父进程中的缓冲区中有数据,因此会复制缓冲区给子进程;同时子进程又执行了一次printf函数,因此子进程打印了两个
-
;同理由i=0创建的子进程也是如此
主要原因:进程在fork时,缓冲区会被复制给子进程,且缓冲区非空。
如果让其只打印6个,printf函数完时,刷新缓冲区
#include<stdio.h>
#include<unistd.h>
int main(int argc,char **argv)
{
for(int i=0;i<2;++i)
{
fork();
printf("*\n");
}
return 0;
}
\n
会在printf函数执行后,刷新printf对应的缓冲区。