今天翻书看到这个,就要分享一下
一般的,fork之后是父进程还是子进程先执行是不确定的。这取决于内核所使用的调度算法。
int glob = 6;
char buf[] = "a write to stdout\n";
int main(void)
{
int var;
pid_t pid;
var = 88;
if(write(STDOUT_FILENO, buf, sizeof(buf)-1) != sizeof(buf) -1)
err_sys("write error");
printf("before fork\n");
if( (pid = fork()) < 0)
err_sys("fork error");
else if(pid == 0){
glob++;
var++;
}else
sleep(2);
printf("pid = %d, glob = %d, var = %d\n", getpid(), glob, var);
exit(0);
}
系统调用write是不带缓存的,因为在fork()之前调用write,所以其数据写到标准输出一次。
但是I/O库函数是带缓存的。当以交互的形式运行改程序(行缓存),只得printf()输出的行一次,因为标准输出缓存由新行符刷新。但是当标准输出重定向到一个文件时,它是全缓存,得到printf()输出行两次。原因:在fork()之前调用了printf一次,但是当调用fork时,该行数据仍在缓存中,然后在父进程数据空间复制到子进程中,该缓冲数据也复制到子进程中。在exit之前,第二个printf将其数据也添加到缓存中。当进程结束时,其缓存内容才会被写到相应的文件中。
下面运行结果:
huwang@huwang:~/demo/unix/chapter08$ ./demo
a write to stdout
before fork
pid = 16016, glob = 7, var = 89
pid = 16015, glob = 6, var = 88
huwang@huwang:~/demo/unix/chapter08$ ./demo > temp
huwang@huwang:~/demo/unix/chapter08$ cat temp
a write to stdout
before fork
pid = 16034, glob = 7, var = 89
before fork
pid = 16033, glob = 6, var = 88
标准I/O提供缓存的目的就是尽可能减少使用read和write调用的数量。不同的缓冲长度,执行I/O所需的CPU时间两有很大的差异。对每个IO流进行缓存管理。一般的,驻在磁盘上的文件通常是由I/O库应用全缓存。当涉及一个终端时,典型的使用行缓存。