fork()与I/O函数

来源:互联网 发布:网络聊天用语段子 编辑:程序博客网 时间:2024/05/16 18:11

今天翻书看到这个,就要分享一下
一般的,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$ ./demoa write to stdoutbefore forkpid = 16016, glob = 7, var = 89pid = 16015, glob = 6, var = 88huwang@huwang:~/demo/unix/chapter08$ ./demo > temphuwang@huwang:~/demo/unix/chapter08$ cat tempa write to stdoutbefore forkpid = 16034, glob = 7, var = 89before forkpid = 16033, glob = 6, var = 88

标准I/O提供缓存的目的就是尽可能减少使用read和write调用的数量。不同的缓冲长度,执行I/O所需的CPU时间两有很大的差异。对每个IO流进行缓存管理。一般的,驻在磁盘上的文件通常是由I/O库应用全缓存。当涉及一个终端时,典型的使用行缓存。

0 0
原创粉丝点击