参考资料:

  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. 4.4MSSQLServer常用版本介绍

    以SQL Server 2008版本为例: -SQL Server 2008 Datacenter(x86 x64 ia64)数据中心版,最强大的版本,要付费的 -SQL Server 2008 En ...

  2. CSS实现漂亮的小水球效果

    先看效果图: 代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> ...

  3. Pyqt5_QmainWindow

    QmainWindow Toolbar() QMenuBar() StatusBar() ******************************************************* ...

  4. Java 在PPT中创建SmartArt图形、读取SmartArt图形中的文本

    一.概述及环境准备 SmartArt 图形通过将文字.图形从多种不同布局.组合来表现内容和观点的逻辑关系,能够快速.有效地传达设计者的意图和信息.这种图文表达的视觉表示形式常用于PPT,Word,Ex ...

  5. dsPIC33EP单片机的PPS(外设引脚选择)

    利用dsPIC33EP单片机进行can通信的时候用到引脚复用 引脚复用通过查询数据手册: C1RX的寄存器为RPINR26.C1RXR=(设置为需要用到的引脚) 引脚设置为输入(C1RX),TRIS= ...

  6. 全网首发,腾讯T3-3整理Netty学习方案(体系图+项目+学习文档)

    前言: 想要学好一门技术,最起码要对他有一定的了解,起码听说过相应的底层原理的东西吧,最起码你要有一点能和别人交流的内容吧,下面是我精简的一点内容,希望对于大家了解netty能有一点帮助 Netty是 ...

  7. webpack@next webpack-multi-page-cli 多页脚手架2.0

    根据自己的经验和想法,对原有的1.x版本进行的大版本的升级.在实际工作中,能结合的应用场景会更加多元化. github:https://github.com/pomelott/webpack-mult ...

  8. pandas删除DataFrame中任意字段等于'null'字符串的行

    删除df中任意字段等于'null'字符串的行: df=df.astype(str)#把df所有元素转为str类型 df=df[df['A'].isin(['null','NULL'])] #找出df的 ...

  9. 通过link的preload进行内容预加载

    Preload 作为一个新的web标准,旨在提高性能和为web开发人员提供更细粒度的加载控制.Preload使开发者能够自定义资源的加载逻辑,且无需忍受基于脚本的资源加载器带来的性能损失. <l ...

  10. vnc远程工具的使用,Windows系统下VNC远程工具的使用教程

    服务器管理工具可以作为VNC的客户端进行VNC的相关操作,是一款功能强大的VNC客户端软件!同时,它也可以作为FTP的客户端,来进行FTP的相关操作!它能够连接Windows和Linux系统下的服务器 ...