概要
-
fork()函数产生的子进程从fork()函数的下一条语句开始执行。
-
标准输入输出是行缓冲,即一行满了才会刷新,那什么是刷新呢?刷新就是将数据从缓冲区取出来,真正能刷新,要满足什么条件呢?
A.满刷新,即一行满了(1024个字节)才会刷新;
B.遇到’\n’会刷新;
C.调用fflush()函数;
D.程序结束 fclose();int p1; printf("AAA\n"); p1=fork();
对于上面的程序段,由于printf()中包含’\n’,使得printf()中的数据从缓冲区输出,因此当fork()创建子进程,将父进程的缓冲区中的数据复制给子进程时,父进程的缓冲区中没有AAA,不会将AAA复制给子进程,子进程也因此不会输出AAA.
反过来,对于代码:
int p1; printf("AAA"); p1=fork();
对于上面的程序段,由于printf()中不包含’\n’,printf()中的数据留在缓冲区,因此当fork()创建子进程,将父进程的缓冲区中的数据复制给子进程时,也会将AAA复制给子进程,子进程因此会输出AAA.
(对于输入输出流和缓冲区更详细的可看这篇文章,对我的帮助很大,开头的行缓冲即取自这篇文章) -
对于程序:
#include <stdio.h> #include <sys/types.h> #include <unistd.h> int main() { int p1; printf("x"); while((p1=fork())==-1); if(p1==0) printf("a"); else printf("b"); printf("y"); return 0; }
输出结果为xbyxay(老师的实验手册说结果是xbyxay或者xayxby,但是我每次的结果都是前者)
但无论如何,子进程输出x不是因为子进程从头到尾又执行了一遍,而是printf(“x”)中不包含\n,x复制给了子进程,当子进程程序结束时,xay一起打印出来。子进程是从if()语句开始执行的,不是从头开始,不然一个fork函数岂不是要创建无数个子进程。
详解
接上面的printf()是否包含‘\n’的问题,程序和对应的结果如下:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
int p1;
printf("AAA\n");
p1=fork();
return 0;
}
结果为:(由于’\n’,子进程未输出AAA)
去掉\n后,代码为:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
int p1;
printf("AAA");
p1=fork();
return 0;
}
结果为:(由于没有’\n’,子进程也输出了AAA)
putchar()也与printf()类似,请看如下程序:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
int p1;
puchar("A");
p1=fork();
return 0;
}
输出为:
结果也很好理解:不满足前面所说的刷新条件,A依然留在缓冲区中,将A复制给子进程,也因此子进程也会输出A
如果,在创建子进程前,将缓冲区中的内容输出,那么A就不会复制给子进程,那么是不是子进程就不会输出A呢?使用如下程序段验证:
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
int p1;
putchar('A');
fflush(stdout);
p1=fork();
return 0;
}
// fflush()的作用是用来刷新缓冲区;
// fflush(stdin)刷新标准输入缓冲区,把输入缓冲区里的东西丢弃;
// fflush(stdout)刷新标准输出缓冲区,把输出缓冲区里的东西强制打印到标准输出设备上
执行结果如下:
如图所见,的确如此。
很多语句的说法不太正确和准确,有错误请指出!