一、文章来由

还是按照惯例来说一下文章为什么来的。晚上好基友在网上刷面试题,看到一个有趣的题目,于是開始了研究,就有了这篇文章。

二、进入正题

题目例如以下:

  1. #include <stdio.h>
  2. #include <sys/types.h>
  3. #include <unistd.h>
  4. int main(void)
  5. {
  6. int i;
  7. for(i=0; i<2; i++){
  8. fork();
  9. printf("-");
  10. }
  11. return 0;
  12. }

绘图理解一下:

假设对fork()的机制比較熟悉的话,上图应该不难理解,数一下超人标志的个数就知道应该是6个

可是,实际上这个程序会非常tricky地输出8个“-”。

下图把”-“改成”s”了

以下開始有借鉴大牛的博客

要讲清这个题。我们首先须要知道fork()系统调用的特性

  1. fork()系统调用是Unix下以自身进程创建子进程的系统调用。一次调用,两次返回。假设返回是0,则是子进程。假设返回值>0,则是父进程(返回值是子进程的pid)。这是众为周知的。

    这个能够有方法巧记一下,就是子进程从fork開始处開始运行。可是此时没有子进程。所以返回值是0;

  2. 另一个非常重要的东西是,在fork()的调用处。整个父进程空间会原模原样地拷贝到子进程中,包含指令,变量值。程序调用栈,环境变量。缓冲区,等等。

所以,上面的那个程序为什么会输入8个“-”,这是由于printf(“-“);语句有buffer。所以,对于上述程序。printf(“-“);把“-”放到了缓存中。并没有真正的输出(參看《C语言的迷题》中的第一题),在fork的时候,缓存被拷贝到了子进程空间。所以,就多了两个,就成了8个,而不是6个。

另外。多说一下。我们知道。Unix下的设备有“块设备”和“字符设备”的概念。所谓块设备。就是以一块一块的数据存取的设备。字符设备是一次存取一个字符的设备。磁盘、内存都是块设备,字符设备如键盘和串口。块设备一般都有缓存。而字符设备一般都没有缓存。

对于上面的问题,我们假设改动一下上面的printf的那条语句为:

  1. printf("-\n");

或是

  1. printf("-");
  2. fflush(stdout);

就没有问题了(就是6个“-”了)。由于程序遇到“\n”,或是EOF,或是缓中区满。或是文件描写叙述符关闭,或是主动flush。或是程序退出,就会把数据刷出缓冲区。

须要注意的是。标准输出是行缓冲。所以遇到“\n”的时候会刷出缓冲区。但对于磁盘这个块设备来说,“\n”并不会引起缓冲区刷出的动作,那是全缓冲,你能够使用setvbuf来设置缓冲区大小。或是用fflush刷缓存。

我预计有些朋友可能对于fork()还不是非常了解,那么我们把上面的程序改成以下这样:

  1. #include <stdio.h>
  2. #include <sys/types.h>
  3. #include <unistd.h>
  4. int main(void)
  5. {
  6. int i;
  7. for(i=0; i<2; i++){
  8. fork();
  9. //注意:以下的printf有“\n”
  10. printf("ppid=%d, pid=%d, i=%d \n", getppid(), getpid(), i);
  11. }
  12. sleep(10); //让进程停留十秒,这样我们能够用pstree查看一下进程树
  13. return 0;
  14. }

于是。上面这段程序会输出以下的结果,(注:编译出的可运行的程序名为fork)

  1. ppid=8858, pid=8518, i=0
  2. ppid=8858, pid=8518, i=1
  3. ppid=8518, pid=8519, i=0
  4. ppid=8518, pid=8519, i=1
  5. ppid=8518, pid=8520, i=1
  6. ppid=8519, pid=8521, i=1
  7. $ pstree -p | grep fork
  8. |-bash(8858)-+-fork(8518)-+-fork(8519)---fork(8521)
  9. | | `-fork(8520)

面对这种图你可能还是看不懂,没事。我好事做究竟,画个图给你看看:

注意:上图中的我用了几个色彩,同样颜色的是同一个进程。

于是,我们的pstree的图示就能够成为以下这个样子:(下图中的颜色与上图相应)

这样,对于printf(“-“);这个语句,我们就能够非常清楚的知道,哪个子进程复制了父进程标准输出缓中区里的的内容,而导致了多次输出了。(例如以下图所看到的,就是我阴影并双边框了那两个子进程)

—END—


从fork面试题開始的思考的更多相关文章

  1. C语言必会面试题(3、耶稣有13个门徒,当中有一个就是出卖耶稣的叛徒,请用排除法找出这位叛徒:13人围坐一圈,从第一个開始报号:1,2,3,1,2,3...。凡是报到“3”就退出圈子,...)

    3.耶稣有13个门徒.当中有一个就是出卖耶稣的叛徒,请用排除法找出这位叛徒:13人围坐一圈,从第一个開始报号:1.2,3.1,2,3.... 凡是报到"3"就退出圈子.最后留在圈子 ...

  2. ExtJs自学教程(1):一切从API開始

    题 记 该系列文章不側重全方位的去介绍ExtJs的使用,仅仅是側重于解决ExtJs问题的思考方法.写的人不用长篇大论,学的人则可以自立更生.l  学习的人仅仅要有一些CSS的javascript的基础 ...

  3. WebGL自学教程——WebGL演示样例:開始

    最终開始WebGL的演示样例了,...... 開始 使用WebGL的步骤,非常easy: 1. 获得WebGL的渲染环境(也叫渲染上下文). 2. 发挥你的想象力,利用<WebGL參考手冊> ...

  4. 关东升的《从零開始学Swift》即将出版

    大家好: 苹果2015WWDC大会公布了Swift2.0,它较之前的版本号Swift1.x有非常大的变化.所以我即将出版<从零開始学Swift><从零開始学Swift>将在&l ...

  5. 从0開始学习 GitHub 系列之「07.GitHub 常见的几种操作」

    之前写了一个 GitHub 系列,反响非常不错,突然发现居然还落下点东西没写,前段时间 GitHub 也改版了,借此机会补充下. 我们都说开源社区最大的魅力是人人多能够參与进去,发挥众人的力量,让一个 ...

  6. 兼容,原来在这里就已经開始--------Day34

    看了两天,算是将w3cschool的javascript部分浏览了一遍.在脑海中大约有了一点概念,也才真切体会到:一入江湖深似海.欲穷此路难上难啊,至少如今看起来是遥遥无期.太多不懂, 太多茫然,只是 ...

  7. [WebGL入门]二,開始WebGL之前,先了解一下canvas

    年2月)HTML5依旧处于草案阶段. HTML5支持网页端的多媒体功能和画布功能,追加了非常多全新的更合理的Tag标签.各个浏览器也都在逐渐的完好这些新的特性. Canvas对象表示一个 HTML画布 ...

  8. 開始Unity3D的学习之旅

    前言:这个系列的文章纯属对自己学习的整理,非高手之作.但确实的记载了我作为一个没接触过3D游戏编程的大学生的心路历程.争取每周整理一次吧.之所以会開始学Unity3D,最基本的原因是由于在快放暑假的时 ...

  9. csdn博客又開始更新了

    csdn博客经过两年多的沉寂又開始更新了,这两年偶尔在http://www.cnblogs.com/JerryWang1991/ 写一些博文,写的也比較少,如今工作一年多了,又開始回到csdn上更新. ...

随机推荐

  1. Python笔记(2)

    Python 一些常用的运算符: 1.算术运算符:+(加).-(减).*(乘)./(除).//(取整).%(取余).**(乘方): 2.比较运算符:>(大于).<(小于).>=(大于 ...

  2. django-4-模板标签,模板继承

    <<<模板标签>>> {% for %}{% endfor %} 循环 {% if %}{% elif %}{% else %}{% endif %} 判断 {% ...

  3. STM32使用HAL库实现ADC单通道转换

    STM32的ADC转换还是很强大的,它具有多个通道选择,这里我就不细说,不了解的可以自行百度,这里只是选取单通道,实现ADC转换.在文章开始之前,我说一下数据左对齐跟右对齐的差别,以前一直糊里糊涂的, ...

  4. Spring 7大功能模块的作用

    1.   Spring 7大功能模块的作用 1)   核心容器(Spring core) 核心容器提供Spring框架的基本功能.Spring以bean的方式组织和管理Java应用中的各个组件及其关系 ...

  5. Java String内存释放

    Java String内存释放 这是一个坑,Java对于String对象,不进行内存的回收: 处理大数据量的时候,少用String. 与JDK有关系:jdk1.6环境下,内存只占用10M,jdk1.8 ...

  6. jQuery——map()函数以及它的java实现

    map()函数小简单介绍 map()函数一直都是我觉得比較有用的函数之中的一个,为什么这么说呢? 先来考虑一下.你是否碰到过下面场景:须要遍历一组对象取出每一个对象的某个属性(比方id)而且用分隔符隔 ...

  7. Redis-2-对于key的通用操作

    Redis-2-对于key的通用操作 标签(空格分隔): redis del key key1 key2 作用: 删除1个或多个键 返回值: 不存在的key忽略掉,返回真正删除的key的数量 rena ...

  8. SharePoint 使用Expression Web 设计网站

    创建好网站以后可就可以开始发布了 possible causes : 1.The web server may not hava the FrontPage Server Extensions ins ...

  9. HTML&CSS——网站注册页面

    1.表单标签 所有需要提交到服务器端的表单项必须使用<form></form>括起来! form 标签属性:  action,整个表单提交的位置(可以是一个页面,也可以是一个后 ...

  10. 安装Windows服务方法

    用sc create 服务名 binPath="路径",不要用老方法InstallUtil会出现一堆的错误