处理输入输出时,我们必须预计到其中可能发生的错误并给出相应的处理措施。
当我们输入时,可能会由于人的失误(错误理解了指令、打字错误等)、文件格式不符、错误估计了情况等原因造成读取失败。
当我们输出时,如果输出设备不可用、队列满或者发生了故障等,都会导致写入失败。
发生输入输出错误的可能情况是无限的!但 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是自定