处理输入输出时,我们必须预计到其中可能发生的错误并给出相应的处理措施。

 当我们输入时,可能会由于人的失误(错误理解了指令、打字错误等)、文件格式不符、错误估计了情况等原因造成读取失败。当我们输出时,如果输出设备不可用、队列满或者发生了故障等,都会导致写入失败。

发生输入输出错误的可能情况是无限的!但 C++ 将所有可能的情况归结为四类,称为流状态(stream state)。每种流状态都用一个 iostate 类型的标志位来表示。

流状态对应的标志位

标志位 意义
badbit 发生了(或许是物理上的)致命性错误,流将不能继续使用。
eofbit 输入结束(文件流的物理结束或用户结束了控制台流输入,例如用户按下了 Ctrl+Z 或 Ctrl+D 组合键。
failbit I/O 操作失败,主要原因是非法数据(例如,试图读取数字时遇到字母)。流可以继续使用,但会设置 failbit 标志。
goodbit 一切止常,没有错误发生,也没有输入结束。

ios_base 类定义了以上四个标志位以及 iostate 类型,但是 ios 类又派生自 ios_base 类,所以可以使用 ios::failbit 代替 ios_base::failbit 以节省输入。

一旦流发生错误,对应的标志位就会被设置,我们可以通过下表列出的函数检测流状态。

C++流状态检测函数及其说明

fail() 和 bad() 之间的区别并未被准确定义。

但是,基本的思想很简单:

 如果输入操作遇到一个简单的格式错误,则使流进入 fail() 状态,也就是假定我们(输入操作的用户)可以从错误中恢复。如果错误真的非常严重,例如发生了磁盘故障,输入操作会使得流进入 bad() 状态。也就是假定面对这种情况你所能做的很有限,只能退出输入。

以上观点导致如下逻辑:

int i = 0;
cin >> i;
if(!cin){  //只有输入操作失败,才会跳转到这里if(cin.bad()){  //流发生严重故障,只能退出函数error("cin is bad!");  //error是自定义函数,它抛出异常,并给出提示信息}if(cin.eof()){  //检测是否读取结束//TODO:}if(cin.fail()){  //流遇到了一些意外情况cin.clear(); //清除/恢复流状态//TODO:}
}

!cin 可以理解为“cin 不成功”或者“cin 发生了某些错误”或者“ cin 的状态不是 good()”, 这与“操作成功”正好相反。

我们在处理 fail() 时所使用的 cin.clear()。当流发生错误时,我们可以进行错误恢复。为了恢复错误,我们显式地将流从 fail() 状态转移到其他状态,从而可以继续从中读取字符。clear() 就起到这样的作用——执行 cin.clear() 后,cin 的状态就变为 good()。

假定我们要读取一个整数序列并存入 vector 中,字符*或“文件尾”表示序列结束。Windows 平台按下 Ctrl+Z 组合键,再按下回车键表示到达文件末尾;类Unix系统按下 Ctrl+D 组合键表示到达文件末尾。

上述功能可通过如下函数来实现:

//从 ist 中读入整数到 v 中,直到遇到 eof() 或终结符
void fill_vector(istream& ist, vector<int>& v, char terminator){for( int i; ist>>i; ) v.push_back(i);//正常情况if(ist.eof()) return;  //发现到了文件尾,正确,返回//发生严重错误,只能退出函数if (ist.bad()){error("cin is bad!");  //error是自定义函数,它抛出异常,并给出提示信息}//发生意外情况if (ist.fail()) {  //最好清除混乱,然后汇报问题ist.clear();  //清除流状态//检测下一个字符是否是终结符char c;ist>>c;  //读入一个符号,希望是终结符if(c != terminator) { // 非终结符ist.unget(); //放回该符号ist.clear(ios_base::failbit);  //将流状态设置为 fail()}}
}

如果发生了 fail(),我们尝试检测下一个字符是否是结束符:如果是,那么就完整得读取了数据,使用 clear() 恢复状态就可以;如果不是,我们就没有办法处理了,所以将状态重新设置为 fail(),以期望 fill_vector() 的调用者(上层函数)有能力处理。

我们通过调用 ist.clear(ios_base::failbit) 来将流状态设置为 fail()。

可以用 unget() 将字符放回 ist,以便 fill_vector() 的调用者可能使用该字符。

如果 fill_vector() 的调用者想知道是什么原因终止了输入,那么可以检测流是处于 fail() 还是 eof() 状态。当然也可以捕获 error() 抛出的 runtime_error 异常,但当 istream 处于 bad() 状态时,继续获取数据是不可能的。

简单起见,可以让 istream 帮我们抛出这个异常。

//当 ist 出现问题时拋出异常
ist.exceptions(ist.exceptions() | ios_base:: badbit);

当语句执行时,如果 ist 处于 bad() 状态,它会抛出一个标准库异常 ios_base::failure。在一个程序中,我们只需要调用 exceptions() 一次。这允许我们简化关联于 ist 的所有输入过程,同时忽略对 bad() 的处理:

//从ist中读入整数到v中,直到遇到eof()或终结符
void fill_vector(istream& ist, vector<int>& v, char terminator){ist.exceptions(ist.exceptions() | ios_base:: badbit);for (int i; ist>>i; ) v.push_back(i);if (ist.eof()) return;  //发现到了文件尾//不是good(),不是bad(),不是eof(),ist的状态一定是fail()ist.clear();  //清除流状态char c;ist>>c;    //读入一个符号,希望是终结符if (c != terminator) { //不是终结符号,一定是失败了ist.unget();    //也许程序调用者可以使用这个符号ist.clear(ios_base::failbit); //将流状态设置为 fail()}
}

这里使用了 ios_base,它是 iostream 的一部分,包含了对常量如 badbit 的定义、异常如 failure 的定义,以及其他一些有用的定义。可以通过::操作符来使用它们,例如 ios_ base::badbit。

与 istream—样,ostream 也有四个状态:good()、fail()、eof() 和 bad()。

C++ 处理输入输出错误相关推荐

  1. C++ 语言学习 day01 (linux ),基本输入输出错误输出函数,名字空间的含义,内联函数,缺省参数,引用 ,、new操作符

    1.介绍c++ C++头文件风格 c++ 老风格:以.h结尾 是c语言风格  stdio.h  (尽量不) c++的转换风格: 去掉.h 在文件名前加c 例如#include <cstdio&g ...

  2. python sys模块 输入输出 错误流

    输出实例: import syssys.stdout.write("msg") # 控制台白色字体打印 普通输出流 sys.stderr.write("msg" ...

  3. python中sys模块有问题_python中sys模块之输入输出错误流

    import sys sys.stdout.write("msg")   # 控制台白色字体打印 普通输出流 sys.stderr.write("msg") # ...

  4. Linux基础到进阶(一)Linux基础命令、输入输出错误重定向、目录分类与作用、vi编辑器

    文章目录 基础命令 输入.输出.错误重定向 输入重定向: < 输出重定向:> 错误重定向 目录分类与作用 vi编辑器 命令模式 底行模式 基础命令 1.cd命令 功能说明:切换目录 cd ...

  5. python怎么查看代码错误_python中的错误如何查看

    python常见的错误有 1.NameError变量名错误 2.IndentationError代码缩进错误 3.AttributeError对象属性错误 4.TypeError类型错误 5.IOEr ...

  6. matlab怎么输入输出文件,[转载]底层文件输入输出函数

    [转载]底层文件输入输出函数 http://sincerewfeng.spaces.live.com/ fclose :关闭文件 fopen :打开文件 fread :从文件中读入二进制数据 fwri ...

  7. python常见错误-python常见的错误提示有什么

    python常见的错误有 1.NameError变量名错误 2.IndentationError代码缩进错误 3.AttributeError对象属性错误 4.TypeError类型错误 5.IOEr ...

  8. python常见错误-python中的错误有什么

    我们在写python代码时,有时候会遇到莫名的错误,而且查阅了很多资料也不明白是怎么回事,尤其是对python初学者而言,会很影响学习的细心,下面总结python中常见的错误,希望帮助到大家. pyt ...

  9. Python 错误和异常小结

    Python异常类  Python是面向对象语言,所以程序抛出的异常也是类.常见的Python异常有以下几个,大家只要大致扫一眼,有个映像,等到编程的时候,相信大家肯定会不只一次跟他们照面(除非你不用 ...

最新文章

  1. 真正的在线教育,开始萌芽了
  2. Java 读写文件大全
  3. mongocollection java_mongodb与java的整合
  4. 构造图 Codeforces Round #236 (Div. 2) C. Searching for Graph
  5. gem ransack(4000✨) 简单介绍
  6. 全志 移除屏幕超时选项 Patch
  7. TCP/IP的分层负载
  8. (03)FPGA发展前景
  9. 基于内容的图像检索概述
  10. ios python3闪退_解决Python3 cx_Freeze打包闪退/无法执行以及No module named 'matplotlib.backends.backend_tkagg'...
  11. java循环语句_Java十四天零基础入门-Java for循环语句
  12. 用springboot+netty搭建在线web聊天室
  13. 三星集团总裁李健熙:72岁老者的争议人生
  14. AI 成野生动物保护神:没有图像识别算法,考拉可能灭绝!
  15. ModSecurity web application firewall (WAF) Research - .Little Hann
  16. 乐器php毕业论文,打击乐器在音乐课堂教学中的应用
  17. 一个硬盘的感人爱情故事
  18. 国产服务器Kylin(aarch64)安装mysql8.0.27
  19. UUID太长怎么办?快来试试NanoId(Java版本)
  20. 计算机2级题无法打开,计算机2级考题word 一  看了必过

热门文章

  1. python中类和对象_Python里的类和对象简介
  2. 类java的步骤_java类加载的过程
  3. 手把手教你dns服务器未响应导致无法上网怎么办
  4. 判断Logger级别是否开启的方法
  5. Ubuntu下virtualenv 安装
  6. 猛将赵云java,这五位三国猛将临危救主,赵云只能排第二位,第一位大家都服...
  7. 2台服务器负载均衡后synchronized_一篇有趣的负载均衡算法实现
  8. php 页面开启错误提示,php开启与关闭错误提示
  9. asp.net接受表单验证格式后再提交数据_看滴普科技大前端如何玩转el-form-renderer 表单渲染器1.14.0
  10. 学习vim的linux游戏,PacVim:一个学习 vim 命令的命令行游戏 | Linux 中国