从多进程函数fork解读linux系统行缓冲与全缓冲
来源:互联网 发布:网络设计方案工程 编辑:程序博客网 时间:2024/06/05 00:16
先说下fork函数:linux系统中,一个现有的进程可以调用fork函数创建一个新的进程。由fork创建的新进程成为子进程(child process)。fork函数被调用一次,但是返回两次。两次返回的唯一区别是子进程的返回值是0,而父进程的返回值则是新子进程的进程ID。子进程和父进程分别继续执行fork调用之后的命令。子进程是父进程的副本(注意是副本,父、子进程不共享资源)。不同的返回值可用于区分当前进程是子进程还是父进程。
记录下书中关于fork的一段代码:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int glob = 6;
int main()
{
int var;
pid_t pid;
var = 88;
var++;
printf("before fork()\n");
if ((pid = fork()) < 0)
{
printf("fork err\n");
}
else if (pid == 0)
{
glob++;
var++;
}
else
{
sleep(2);
}
printf("pid = %d, glob = %d, var = %d\n", getpid(), glob, var);
return 0;
}
这段代码不是很难,很容易理解,难以理解的是它的输出:
[root@isayme fork]# ./a.out
before fork()
pid = 3385, glob = 7, var = 90
pid = 3384, glob = 6, var = 89
[root@isayme fork]# ./a.out > rel.txt
[root@isayme fork]# cat rel.txt
before fork()
pid = 3404, glob = 7, var = 90
before fork()
pid = 3403, glob = 6, var = 89
不同的方式执行程序,生成的结果不同。原因就在于行缓冲与全缓冲!先简单说下这两个缓冲的概念:
行缓冲:在这种情况下,当输入或输出缓冲区中遇到换行符时,标准I/O库执行I/O操作,输出信息便会显示出来。
全缓冲:在这种情况下,只有当输入或输出缓冲区满时,标准I/O库才会执行I/O操作,输出信息才显示出来。
明白了缓冲的概念,我们可以解释上面的问题:
首先解释第一个输出的原因:第一次执行的方式,因为数据写到标准输出,所以是行缓冲。执行printf("before fork()\n");语句时将直接输出信息,因为遇到了换行符;然后执行fork函数,正常情况是不会fork失败,除非系统存在的进程确实太多了,fork成功后就同时存在父子进程,由于系统的调度算法的复杂,下面的代码执行的顺序可能不是那么严格的顺序,为了确保顺序,在父进程中sleep了2秒,确保子进程中最后的输出代码先执行。所以输出的部分是子进程的输出信息在前,父进程的在后。在子进程中,glob和var两个变量都进行了加1操作,所以值响应的比父进程的值大1。
现在说下第二个输出的原因:第二次执行的方式,引文数据写到文件中,所以是全缓冲,遇到换行符并不输出信息,而是将输出信息存在缓冲区中,最后一次性输出(本程序的输出信息明显不足以让输出缓冲区满)。因为子进程是父进程的副本,其中包括输出缓冲区副本,此时printf("before fork()\n");输出的信息在父子进程的输出缓冲区中都存在,由于父进程在执行过程中sleep了2秒,所以子进程先结束,进程退出前会将输出缓冲区中的信息写到标准输出设备,即我们所见到的屏幕,所以先输出的信息中glob和var的值都是加1后的值。而后父进程也结束并将其的输出缓冲区的信息输出到屏幕,由于父子进程的输出缓冲区中都存在printf("before fork()\n");代码段的输出信息,所以此种执行方式的输出结果中存在两次"printf("before fork()\n");"。
PS:flush函数可以强制将输出缓冲区的额内容输出到屏幕,有兴趣的可以自行测试在代码中加入此函数后的输出结果。
转载自: 独语者
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("hello\n");
fork();
printf("world\n");
fork();
printf("nice \n");
}
用gcc编译之后的运行结果如下:
hello
world
nice
nice
world
nice
nice
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("hello");
fork();
printf("world\n");
fork();
printf("nice ");
}
helloworld
nice nice helloworld
nice nice
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
printf("hello");
fork();
printf("world");
fork();
printf("nice ");
}
helloworldnice helloworldnice helloworldnice helloworldnice
- 从多进程函数fork解读linux系统行缓冲与全缓冲
- 通过fork函数解读linux系统行缓冲与全缓冲
- 从一个fork()实例理解全缓冲与行缓冲
- 从一个fork()实例理解全缓冲与行缓冲
- fork与行缓冲
- 全缓冲与行缓冲
- linux下fork()函数在行缓冲和全缓冲情况下对IO的影响
- fork子进程缓冲
- fork举例和行缓冲全缓冲的理解
- Linux 标准IO库 全缓冲、行缓冲、无缓冲
- 行缓冲、全缓冲、无缓冲
- 全缓冲和行缓冲
- 标准IO缓冲详解 全缓冲 行缓冲 不缓冲
- 标准I/O缓冲:全缓冲、行缓冲、无缓冲
- 标准I/O缓冲:全缓冲、行缓冲、无缓冲
- 标准IO缓冲详解全缓冲、行缓冲、不缓冲
- 标准I/O缓冲:全缓冲、行缓冲、无缓冲 .
- 标准IO缓冲详解全缓冲、行缓冲、不缓冲
- 流年轻弹、离殇一曲
- C++库函数使用
- 给大一师弟师妹的一些建议
- 字符串匹配(String Matching)
- 演示matplotlib中,如何在坐标系中画一个矩形
- 从多进程函数fork解读linux系统行缓冲与全缓冲
- 【贪心】 hdu4415 Assassin’s Creed
- 安装虚拟机后锐捷提示把其他的网卡禁用
- 学习方法
- ASP.NET小技巧——回传后保持页面的滚动位置
- 等你,执手渡流年(七夕)
- JAVA发送邮件和获取客户端真实ip地址
- proxy software with bugs solution
- linux 内核源代码目录结构