最近看书时又搞到输入流了,以前感觉还没什么,但自己一写时才发现,啥都不对,神马问题都来了,晕倒,算了,安心的把输入流理解理解,测试结果及代码搞上:

 

首先看看cin.get(),它是一个读取单个字符的方法.

字符变量=cin.get();相当于cin.get(字符变量);

 

cin的输出返回一个cin对象,如果输入错误返回一个bool值,所以可以用if(cin)来判断输入有没有错,if(cin)如果为真就说明输入正确,否则错误。

 

int main()

{

char c;

c=cin.get();    //读取单个字符,在屏幕输入,也相当于cin.get(c);

cout<<c<<endl;  //输出刚刚载入的单个字符

}

运行程序后,一切正常:

输入:a    输出:a

但当我们输入的不只一个英文字符时,那又会如何呢?

输入:abcd         输出:a

由此可知,它只能读取第一个字符,但如果我们把程序修改成:

int main()

{

char c;

char b;

c=cin.get();        //读取单个字符,在屏幕输入

b=cin.get();  

cout<<c<<b<<endl;  //输出刚刚载入的单个字符

}

我们再输入:abcd    最后输出了:ab

既然cin.get()是读取第一个字符,那b为什么不也是a呢?

其实是这样的:

在cin这个对象里,有一个储存字符的流,可以想象成缓冲区,但事实上是cin里封装的一个东西.当我们在程序上输入字符后,对象cin获得了我们输入的 字符,例如获得abcd,然后再通过.get()把流里面的第一个字符去掉,赋给c,这时,cin里储存的流的数据为bcd,而cstr则获得了 a.当我们再次运行b=cin.get();时,同理把cin里流的数据的b拿出来给了bstr,此后,cin里面的流的数据为cd,而bstr则 为b,所以最后输出时,便能输出ab了.

还有个补充,究竟什么时候才输入数据呢?我们可以再通过上面的代码进行尝试,我们输入单个字母'a',然后按回车,发现并没有输出数据,而是再等待 一次输入数据,我们再输入字母'b',按回车后便输出ab了.相信到这里,大家都应该明白了,因为当我们第一次输入a后,通过 cstr=cin.get();使cin里的流没有数据,清空了.所以到第二次要再赋给b值时,它找不到数据,要重新再输入数据.由此来看可以知 道,当cin里的流数据清空时,便需要重新输入才能赋值.而cin.get()还有个用法:

int main()

{

char c;

char b;

c=cin.get();        //读取单个字符,在屏幕输入

  cin.get();        //忽略一个字符或者删除一个字符。

b=cin.get();  

cout<<c<<b<<endl;  //输出刚刚载入的单个字符

}

 

程序中有3个cin.get(),所以我们尝试输入:abc.   发现输出了:ac

由此能知道,当空回调cin.get();时,cin.get便自动在cin中的流数据中删除一个字母,起了一个删除作用.

对cin.get()有了一定了解之后,对cin.getline()的学习就可以更快了,原理是一致的,但是cin.getline()则是获取一整行文本.以下是cin.getline()原形:

getline(char *line,int size,char='\n')

第一个就是字符指针,第二个是字符长度,第三个1行的结束标识符.

int main()

{

char cstr[200];

cin.getline(cstr,sizeof(str));     //第三个参数不输入,默认回车为结束标识符

cout<<cstr<<endl;                //输出

}

这样我们输入一堆英文或数字,然后按回车,就会输出一行刚刚输出的东西了.接下来.我们讨论第三个参数的作用.

int main()

{

char cstr[200];

cin.getline(cstr,sizeof(str),'X');  //我们以单个英文字母'X'作为终止标识符

cout<<cstr<<endl;                   //输出

}

当我们输入一大堆东西,例如

输入: kkkkkkk(回车)              输出: kkkkkkk(回车)                        

bbbbbbb(回车)                     bbbbbbb(回车)       

lllllX                                          lllll

这样X便成了终止符,其原理和cin.get一样.

接下来我们谈谈cin.clear的作用,第一次看到这东西,很多人以为就是清空cin里面的数据流,而实际上却与此相差很远,首先我们看看以下代码:

int main()

{

int a;

cin>>a;

cout<<cin.rdstate()<<endl;

if(cin.rdstate() == ios::goodbit)

{

cout<<"无错误!"<<endl;

}

if(cin.rdstate() == ios_base::failbit)

{

cout<<"非致命错误,可清除输入缓冲区挽回!"<<endl;

}

system("pause");

}

我们定义要输入到的变量是整型,但如果我们输入了英文字母或者汉字,那就会发生错误,cin里有个方法能检测这个错误,就是cin.rdstate();

当cin.rdstate()返回0(即ios::goodbit)时表示无错误,可以继续输入或者操作,若返回4则发生非致命错误即ios::failbit,则不能继续输入或操作.而cin.clear则可以控制我们此时cin里对这个问题的一个标识.语发如下:

cin.clear(标识符);

标识符号为:

goodbit 无错误

Eofbit 已到达文件尾

failbit 非致命的输入/输出错误,可挽回

badbit 致命的输入/输出错误,无法挽回

若在输入输出类里.需要加ios::标识符号

通过cin.clear,我们能确认它的内部标识符,如果输入错误则能重新输入.结合真正的清空数据流方法cin.sync(),请看下例:

int main()

{

int a;

while(1)

{

cin>>a;

if(!cin)            //条件可改写为cin.fail()

{

cout<<"输入有错!请重新输入"<<endl;

cin.clear();

cin.sync();   //清空流

}

else

{

cout<<a;

break;

}

}

}

上面的cin默认参数为1,即无错误,正常操作.当我们输入英文字母'k'时,它的状态标识改为0,即错误,用cout对用户输出信 息,再用cin.clear让错误标识改回为0,让我们可以继续输入,再清空流数据继续输入.如果我们没有了cin.clear,则会进入死循环,其过程 为我们输入了英文字母,它的状态标识便为fail,当运行到条件判断时,便总是回到错误的条件表示里,并且我们再也没办法输入,因为错误的表示关闭了 cin,所以会进入死循环.