参考资料:

  https://www.douban.com/group/topic/127062773/

  今天学习了C++语言的标准I/O,也就是std::cin和std::cout,但是我发现当系统在读取标准的输入后需要按两次ctrl+D或者按一次回车再按一次ctrl+D才能结束标准I/O,翻阅相关资料后我把这个问题研究透彻了记录在此。(使用类Unix系统,所以EOF是ctrl+D,windows上可能是ctrl+Z)

  首先我们必须知道一个概念:缓冲区,缓冲区是干什么的?


  我们为什么要引入缓冲区?

  比如我们从磁盘里取信息,我们先把读出的数据放在缓冲区,计算机再直接从缓冲区中取数据,等缓冲区的数据取完后再去磁盘中读取,这样就可以减少磁盘的读写次数,再加上计算机对缓冲区的操作大大快于对磁盘的操作,故应用缓冲区可大大提高计算机的运行速度。

  又比如,我们使用打印机打印文档,由于打印机的打印速度相对较慢,我们先把文档输出到打印机相应的缓冲区,打印机再自行逐步打印,这时我们的CPU可以处理别的事情。现在您基本明白了吧,缓冲区就是一块内存区,它用在输入输出设备和CPU之间,用来缓存数据。它使得低速的输入输出设备和高速的CPU能够协调工作,避免低速的输入输出设备占用CPU,解放出CPU,使其能够高效率工作。  


  缓冲区的种类?

  1. 全缓冲:在这种情况下,当填满标准I/O缓存后才进行实际I/O操作。全缓冲的典型代表是对磁盘文件的读写。
  2. 行缓冲:在这种情况下,当在输入和输出中遇到换行符时,执行真正的I/O操作。这时,我们输入的字符先存放在缓冲区,等按下回车键换行时才进行实际的I/O操作。典型代表是键盘输入数据。
  3. 不带缓冲:也就是不进行缓冲,标准出错情况stderr是典型代表,这使得出错信息可以直接尽快地显示出来。

  我们研究的对象是键盘的输入,也就是行缓冲了,下面用一个例子说明问题:

#include <iostream>

/*
* Simple main function:
* Read several numbers and write their sum
*/
int main()
{
int sum = 0, val = 1;
while(std::cin >> val){
sum += val;
}
std::cout << "The sum is " << sum << std::endl;
return 0;
}

  在这个例子中我们从标准输入中取值,赋给val,当赋给val的值是可用的值时,while循环继续,否则打破循环。

  读者可以试验一下这两种情况:

  1.运行程序之后输入1 2 3然后按回车键,此时程序不结束,再按ctrl+D键,也就是输入EOF,程序给出输出6。

  2.运行程序之后输入1 2 3 (注意3后面我多输入一个空格,上面那个例子我不输入)然后按ctrl+D,没反应,再按ctrl+D,程序输出6。

  我们用学到的缓冲区的知识细致细致再细致地分析这个过程:

  第一种情况:

  输入1 2 3,此时缓冲区是这样的:'1' '空格' '2' '空格' '3',因为行缓冲,输入回车后'1' '空格' '2' '空格' '3' '回车'被送给CPU,也就是送给while了,while拿到数据后很高兴,按照空格和回车都是分割符的原理,将1 2 3交给val加了起来。此时缓冲区呢:空了!!但是程序还在期待输入,但我不想再输入了,那么在一个空的缓冲区中输入ctrl+D就可以直接结束标准输入。

  第二种情况:

  输入1 2 3 (3后面有空格),此时缓冲区是这样的:'1' '空格' '2' '空格' '3' '空格',根据EOF符号的定义,此时我们打出EOF符号(ctrl+D)。'1' '空格' '2' '空格' '3' '空格'被送给CPU,同时EOF被丢弃了。while拿到数据后又很高兴,按照空格是分割符的原理,将1 2 3交给val加了起来。但是输入还没完啊,此时'1' '空格' '2' '空格' '3' '空格'交给CPU了,EOF被丢弃了,缓冲区是不是空了?此时再打出EOF符号,标准输入即被关闭。

  读者可能会问为什么第二种情况在3后面要加上空格,是这样的,可以考虑不加空格是什么样的结果:CPU会拿到'1' '空格' '2' '空格' '3',此时因为3后面没有分隔符,那么我们此时输入EOF就是把标准输入截断了。假如我们输入'8' '空格',再按两次ctrl+D,此时sum的加和是1+2+38=41,所以说这种情况就和std::cin传值机理的问题了,和缓冲区的原理关系不大。

  总结一下:

  回车键的效果:将缓冲区连带自身传给CPU,缓冲区清空。

  ctrl+D(EOF)的效果:将缓冲区不带自身传给CPU,缓冲区清空。

  缓冲区为空时收到EOF,标准文件输入关闭!

C++中为什么按两次ctrl+D才能结束标准I/O的更多相关文章

  1. 浏览器缓存如何控制? && 在url框中回车、F5 和 Ctrl + F5的区别是什么?

    第一部分: 浏览器缓存如何控制?   最近在做网站,但是不知道缓存是什么东西怎么能行! 如何实现HTTP缓存呢? 下面我们来一步一步的探寻实现机制把. 方案一: 无缓存   说明: 浏览器向服务器请求 ...

  2. 浏览器缓存控制 以及 在url框中回车、F5 和 Ctrl + F5的区别

    第一部分: 浏览器缓存如何控制?   做网站,不知道缓存是什么东西怎么能行! 如何实现HTTP缓存呢? 下面我们来一步一步的探寻实现机制把. 方案一: 无缓存   说明:浏览器向服务器请求资源m.pn ...

  3. Android中Fragment的两种创建方式

    fragment是Activity中用户界面的一个行为或者是一部分.你可以在一个单独的Activity上把多个Fragment组合成为一个多区域的UI,并且可以在多个Activity中再使用.你可以认 ...

  4. 【跟着子迟品 underscore】JavaScript 中如何判断两个元素是否 "相同"

    Why underscore 最近开始看 underscore.js 源码,并将 underscore.js 源码解读 放在了我的 2016 计划中. 阅读一些著名框架类库的源码,就好像和一个个大师对 ...

  5. [Linux]在linux中,常常用到ctrl和其他按键组合,常用的有哪些及意义呢

    在linux中,常常用到ctrl和其他按键组合,常用的有哪些及意义呢? Ctrl+c 结束正在运行的程序 Ctrl+d 结束输入或退出shell Ctrl+s 暂停屏幕输出[锁住终端] Ctrl+q ...

  6. java 中 return 的两种常见的用法

    一:return语句总是用在方法中,有两个作用: 一个是返回方法指定类型的值(这个值总是确定的), 一个是结束方法的执行(仅仅一个return语句). 二:实例1 -- 返回一个String priv ...

  7. java中线程分两种,守护线程和用户线程。

    java中线程分为两种类型:用户线程和守护线程. 通过Thread.setDaemon(false)设置为用户线程: 通过Thread.setDaemon(true)设置为守护线程. 如果不设置次属性 ...

  8. Android向系统相册中插入图片,相册中会出现两张 一样的图片(只是图片大小不一致)

    向系统相册中插入图片调用此方法时,相册中会出现两张一样的图片 MediaStore.Images.Media.insertImage 一张图片是原图一张图片是缩略图.表现形式为:android4.4. ...

  9. 处理Selection对象和Range对象——Word VBA中重要的两个对象

    处理Selection对象和Range对象——Word VBA中重要的两个对象 Word 开发人员参考Selection 对象代表窗口或窗格中的当前所选内容.所选内容代表文档中选定(或突出显示)的区域 ...

随机推荐

  1. 【Mac】屏蔽系统升级更新

    三行代码解决Mac升级弹窗,小红点数字1

  2. 【python爬虫】解决歌荒,下歌利器

    python下载图片,mp3,想必很多人都早已耳闻,今天给大家来点不一样的, 让你下载高逼格高品质,带进度条,实时显示下载速度 详见源码:https://www.kesci.com/home/proj ...

  3. OpenResty高性能web平台

    openresty高性能web平台安装使用 简介:OpenResty® 是一个基于 Nginx 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库.第三方模块以及大多数的依赖项.用 ...

  4. SpringBoot系列——状态机(附完整源码)

    1. 简单介绍状态机 2. 状态机的本质 3. 状态机应用场景 1. 简单介绍状态机 状态机由状态寄存器和组合逻辑电路构成,能够根据控制信号按照预先设定的状态进行状态转移,是协调相关信号动作.完成特定 ...

  5. 使用websocket开发智能聊天机器人

    前面我们学习了异步web框架(sanic)和http异步调用库httpx,今天我们学习websocket技术. websocket简介 我们知道HTTP协议是:请求->响应,如果没有响应就一直等 ...

  6. 括号树 noip(csp??) 2019 洛谷 P5658

    洛谷AC通道 本题,题目长,但是实际想起来十分简单. 首先,对于树上的每一个后括号,我们很容易知道,他的贡献值等于上一个后括号的贡献值 + 1.(当然,前提是要有人跟他匹配,毕竟题目中要求了,是不同的 ...

  7. 初窥 BB-Framework

     

  8. C#线程 使用线程

    第三部分 使用线程 基于事件的异步模式 基于事件的异步模式(EAP)提供了一种简单的方法,通过这些方法,类可以提供多线程功能,而使用者无需显式启动或管理线程.它还提供以下功能: 合作取消模型 工作人员 ...

  9. PriorityBlockingQueue 和 Executors.newCachedThreadPool()

    1.PriorityBlockingQueue里面存储的对象必须是实现Comparable接口. 2.队列通过这个接口的compare方法确定对象的优先级priority. 规则是:当前和其他对象比较 ...

  10. (Java实现) 洛谷 P1115 最大子段和

    题目描述 给出一段序列,选出其中连续且非空的一段使得这段和最大. 输入输出格式 输入格式: 第一行是一个正整数NN,表示了序列的长度. 第二行包含NN个绝对值不大于1000010000的整数A_iA ...