关于if (!cin)以及while (cin >> word)
首先注意:
1、每一个输入(输出)对象就代表一个输入(输出)流;
2、输入(输出)对象中的流状态成员标记了输入(输出)流当前的状况,当eofbit、badbit、failbit三个标记位均为0时表示流状态正常;
3、一但某个或几个标记位被设置,表示对象的流状态出现相应状况,流将对后面的输入(输出)关闭,直到标记位被清除;
4、只有在流状态良好的情况下,if或者while对该输入(输出)对象的判断才能是ture。
接下来看看这个程序:
#include <iostream>
using namespace std; int main()
{
int input;
while (cin >> input)
{
cout << input << endl;
cout << "rdstate()函数的返回值: " << cin.rdstate() << endl;
cout << "三个标记位返回的返回值: " << cin.bad() << cin.fail() << cin.eof() << endl;
cout << "goodbit标记位的返回值: " << cin.good() << endl;
cout << endl;
} cout << "rdstate()函数的返回值: " << cin.rdstate() << endl;
cout << "三个标记位返回的返回值: " << cin.bad() << cin.fail() << cin.eof() << endl;
cout << "goodbit标记位的返回值: " << cin.good() << endl; if (!cin)
cout << "input error" << endl; else
cout << input << endl; system ("pause");
return ;
}
当输入为:“123 124 158 137a”时,输出为:

具体分析如下:
123、124、158均被正常抽取,从rdstate()等 流状态标记位测试函数的返回结果可以看出来。
当轮到137a进入程序时,抽取操作符将137成功抽取,将a留在输入流中。抽取操作符“ >> ”返回cin对象,此时流状态仍为正常,failbit、eofbit、badbit三个标记位的值均为0。然后while判定cin对象为true,使得进入循环输出137及流状态信息。
接下来进入下一次循环,执行 cin >> input ; 语句:a为输入流的第一个字符,a与input变量类型不匹配,导致cin流状态的failbit标记位被置为1(goodbit标记位被置为0),这使得“ >> ”返回的cin对象自身检测为False, 然后while判定cin对象为false,退出循环。而字符a,留在了输入流中。
退出循环后输出cin的流状态,可以看出,rdstate()的结果是4,failbit标记位为1,goodbit标记位为0。
程序流程到了 if 语句,此时cin对象的failbit标记位已经被置为1(goodbit标记位被置0),if 判定cin对象为false ,那么 !cin 为true,因此输出input error。
当输入文件结束符(^Z,即Ctrl+Z 或 F6)时,输出为:

badbit 标志着系统级的故障,如无法恢复的读写错误。如果出现了这类错误,则该流通常就不能再继续使用了。如果出现的是可恢复的错误,如在希望获得数值型数据时输入了字符,此时则设置 failbit 标志,这种导致设置 failbit的问题通常是可以修正的。eofbit 是在遇到文件结束符时设置的,此时同时还设置了 failbit。
流的状态由 bad、fail、eof 和 good 操作提示。如果 bad、fail 或者 eof中的任意一个为 true,则检查流本身将显示该流处于错误状态。类似地,如果这三个条件没有一个为 true,则 good 操作将返回 true。
再来看一看下一段代码:
if (cin >> input)
cout << input << endl;
else
cout << "input error" << endl; if (!cin)
cout << "input error" << endl;
else
cout << input << endl;
当输入为567a时,输出为:
567
567
原因:输入流中的567被“ >> ”抽取到 input 中,因此抽取操作符“ >> ”返回的cin对象的流状态成员正常。因此 !cin 为false,使得再一次输出567。注意:字符a留在了输入流。
但当代码重复一次时:
if (cin >> input)
cout << input << endl;
else
cout << "input error" << endl; if (!cin)
cout << "input error" << endl;
else
cout << input << endl;
//第一次代码结束 if (cin >> input)
cout << input << endl;
else
cout << "input error" << endl; if (!cin)
cout << "input error" << endl;
else
cout << input << endl;
//重复代码结束
如果输入567a,输出是:
567
567
input error
input error
原因是:当第一次代码正常完成后, 第二次代码开始,输入流中的第一个字符是a,a与input不匹配,“ >> ”无法抽取, 抽取操作符“ >> ”函数返回的cin对象,其failbit标记位被置为1(goodbit标记位被置为0) ,cin的自测为False,if 判定条件为false;且导致下一个if语句: if (!cin) 中, cin对象为false, !cin 为true。
真相是:只要抽取操作符“ >> ”能成功抽取输入流中的数据,哪怕只是部分匹配,cin对象的流状态成员(即标记位)不会被设置,流状态仍将保持正常。
一旦输入完全不匹配,“ >> ”无法进行任何抽取,返回的cin对象其failbit标记位置为1(goodbit标记位被置0),cin流状态出现相应状况 ,且在恢复之前,流将对后面的输入关闭。
不对之处,敬请指教。
参考文献:http://blog.csdn.net/ygj149078299/article/details/538998
关于if (!cin)以及while (cin >> word)的更多相关文章
- [原创]cin、cin.get()、cin.getline()、getline()、gets()、getchar()的区别
这几个输入函数经常搞不清具体特点和用法,这里稍作总结 一.cin>> 1.最基本用法,输入一个变量值 2.输入字符串,遇“空格”.“TAB”.“回车”结束,比如输入“hello world ...
- C++中关于cin、cin.get()、cin.getline()、getline()、gets()等函数的用法
1.cin>> 用法1:最基本,也是最常用的用法,输入一个数字: 注意:>> 是会过滤掉不可见的字符(如 空格 回车,TAB 等) cin>>noskipws> ...
- C++中cin、cin.get()、cin.getline()、getline()、gets()等函数的用法----细节决定成败 (sort用法)
C++中cin.cin.get().cin.getline().getline().gets()等函数的用法 学C++的时候,这几个输入函数弄的有点迷糊:这里做个小结,为了自己复习,也希望对后来者能有 ...
- C++中cin、cin.get()、cin.getline()、getline()、gets()等函数的用法
学C++的时候,这几个输入函数弄的有点迷糊:这里做个小结,为了自己复习,也希望对后来者能有所帮助,如果有差错的地方还请各位多多指教(本文所有程序均通过VC 6.0运行) 1.cin 2.cin.get ...
- cin、cin.get()、cin.getline()、getline()、gets()等函数的用法
学C++的时候,这几个输入函数弄的有点迷糊:这里做个小结,为了自己复习,也希望对后来者能有所帮助,如果有差错的地方还请各位多多指教(本文所有程序均通过VC 6.0运行)转载请保留作者信息:1.cin1 ...
- [转载]cin、cin.get()、cin.getline()、getline()、gets()函数的用法
1.cin>> 用法1:最基本,也是最常用的用法,输入一个数字: #include <iostream>using namespace std;main ( ...
- (转)cin、cin.get()、cin.getline()、getline()、gets()等函数的用法
学C++的时候,这几个输入函数弄的有点迷糊:这里做个小结,为了自己复习,也希望对后来者能有所帮助,如果有差错的地方还请各位多多指教(本文所有程序均通过VC 6.0运行)转载请保留作者信息:1.cin1 ...
- cin详解(cin.get()、cin.getline()、cin.clear()、cin.sync())
在C中,输入输出用scanf和printf,在输入数据的同时还需说明数据的类型,如果输入数据较多,那就很麻烦,而C++中也有相似的东西cin和cout,它们来自C++的一个名叫" iostr ...
- C++中几个输入函数的用法和区别(cin、cin.get()、cin.getline()、getline()、gets()、getchar())
1.cin>> 用法1:最基本,也是最常用的用法,输入一个数字: #include <iostream>using namespace std;main (){int a,b; ...
随机推荐
- smb相关资料
smb相关资料 看资料就上维基 https://en.wikipedia.org/wiki/Server_Message_Block#Implementation http://www.bing.co ...
- .NET与你若仅仅如初见(一)
难忘初次见到你,那是一个夏日的午后,可是天空中乌云密布.大雨来临前的一段时间总是非常闷热的,当我朦胧的睡眼看到你之后瞬间就清醒了,感觉空气也凉爽了起来.尽管仅仅一眼但就是被你那清新脱俗沉鱼落雁之美所征 ...
- caffe源代码分析--data_layer.cpp
dataLayer作为整个网络的输入层, 数据从leveldb中取. leveldb的数据是通过图片转换过来的. 网络建立的时候. datalayer主要是负责设置一些參数,比方batchsize.c ...
- Codeforces 482B Interesting Array(线段树)
题目链接:Codeforces 482B Interesting Array 题目大意:给定一个长度为N的数组,如今有M个限制,每一个限制有l,r,q,表示从a[l]~a[r]取且后的数一定为q,问是 ...
- 让你提前认识软件开发(23):怎样在C语言中运行shell命令?
第1部分 又一次认识C语言 怎样在C语言中运行shell命令? [文章摘要] Linux操作系统具备开源等诸多优秀特性,因此在很多通信类软件(主流开发语言为C语言)中,开发平台都迁移到了Linux上, ...
- IOS中的几中观察监听模式
本文介绍Objective C中实现观察者模式(也被称为广播者/监听者.发布/注册或者通知)的五种方法以及每种方法的价值所在. 该文章将包括: 1 手动广播者和监听者(Broadcaster and ...
- ajax封装
/** * ITCAST WEB * Created by zhousg on 2016/5/24. */ /* * 1. 请求的类型 type get post * 2. 请求地址 url * 3. ...
- MSMQ是什么?
MSMQ(MicroSoft Message Queue,微软消息队列)是在多个不同的应用之间实现相互通信的一种异步传输模式,相互通信的应用可以分布于同一台机器上,也可以分布于相连的网络空间中的任一位 ...
- 【转】Android的线程使用来更新UI----Thread、Handler、Looper、TimerTask
方法一:(java习惯,在android不推荐使用) 刚刚开始接触android线程编程的时候,习惯好像java一样,试图用下面的代码解决问题 new Thread( new Runnable() { ...
- android开发网络连接工具类(一)
网络连接工具类整理: package com.gzcivil.utils; import java.io.IOException; import java.util.ArrayList; import ...