C++ 之cin与cout常见问题详解

经常要使用到cin和cout,花了点时间,把常见的问题整理了一下,供记性不好时查阅。

在程序中包含iostream文件将自动创建cin对象和cou对象,即标准输入流和标准输出流。


关于cout :

<<操作符的默认含义是按位左移操作符,但ostream类重新定义了<<操作符,将其重载为输出方法。<<支持C++的所有的基本类型。对于每一种数据类型,C++都提供了operator<<()的定义。例如,如果表达式cout << 100对应于下面的方法:

Ostream & operator<<(int);

Ostream类还为以下的指针类型定义了插入操作符函数:

Const signed char*;

Const unsigned char*;

Const char*;

Void *;

Cout 支持拼接输出:

如: cout << "hello" <<"world"<


怎样刷新输出缓冲区?

由于ostreamcout对象处理的输出进行缓冲,所以输出不会立即发送到目的地址,而是被存储在缓冲区中,直到缓冲区填满。然后,程序将刷新缓冲区,把内容发送出去,并清空缓冲区,以存储新的数据。通常,缓冲区为512字节或其整数倍。

有两种方式实现缓冲区的手动刷新:

Flush:刷新缓冲区

Endl:刷新缓冲区,并插入一个换行符

例如: 

  cout << "hello world" << flush;

  Cout << "hello world" <


关于cin:

Cin对象将标准输入表示为字节流。输入可以是字符串的一部分、int值、float值等,istreaml类重载了抽取操作符>>,使之能够识别C++所有的基本数据类型,所以,抽取还涉及到类型转换。编译器将根据数据接收者的类型,匹配适当的的函数。

例如: cin >> value_holder 

如果value_holderint型,则将匹配函数

istream & operator>>(int &); //注意函数原型,参数和返回值都是引用

而如果value_holderchar类型,则将匹配函数

istream & operator >>(char &);

istream类为下列字符指针类型重载了>>抽取操作符:

signed char*;

char *;

unsigned char *;

Cin的常见用法:

用法1:最基本,也是最常用的用法,输入一个数字:

[java]  view plain copy
  1. #include   
  2.  using namespace std;  
  3.  int main () {  
  4.  int a,b;  
  5.  cin>>a>>b;   
  6.  cout<
  7.  return 0;  
  8. }  

注意:>> 是会过滤掉不可见字符(如 空格 回车,TAB 等) 
cin>>noskipws>>input[j];//不想略过空白字符,那就使用 noskipws 流控制

用法2:接受一个字符串,遇空格“TAB”回车都结束

[java]  view plain copy
  1. #include    
  2. using namespace std;   
  3. int main () {   
  4.  char a[20];  
  5.  cin>>a;   
  6.  cout<
  7. return 0  
  8. }  

Cin有一个比较“智能”之处,即它会读取从非空白字符开始,到与目标类型不匹配的第一个字符之间的全部内容。

例如,对于下面的代码:

Int value;

Cin >> -123a;

操作符将提取-,1,2,3,因为它们都是整数的有效部分。但是a字符不是有效字符,因此输入中最后一个可接受的字符是3a将留在输入流中,下一个cin语句将从这里开始读取。与此同时,操作符将字符序列-123转换为一个整数值,并将它赋给value


怎样实现循环输入?

先看这样一段代码:

[java]  view plain copy
  1. int sum =0;  
  2. int input;  
  3. While(cin >> input){  
  4.     Sum += input;  
  5. }  

这段代码实现了循环输入,循环体中当然可以用别的数据结构比如容器来接收输入,这不是一个死循环吗?当然,如果一直输入的是正确的int型数据,输入将一直进行下去,直到地老天荒!结束这个循环的方法很简单,输入一个错误字符即可。

cin操作未能读取到预期的字符时,它将设置流状态failbit.,此时,if或者else将断定cin对象为false


怎样恢复流状态?

>>提取到非预期字符时,会设置failbit,此时流会关闭。如果要继续输入,必须要重置流状态,同时还要清空流缓存中遗留的数据。

Cin.clear();//重置流状态

Cin.sync();//清空流缓存

这两个函数要同时使用才算完成重置流状态。

以下代码说明sync()函数的功能:

[java]  view plain copy
  1. #include   
  2. using namespace std;  
  3.   
  4. int main () {  
  5.   char first, second;  
  6.   
  7.   cout << "Please, enter a word: ";  
  8.   first=cin.get();  
  9.   cin.sync();  
  10.   
  11.   cout << "Please, enter another word: ";  
  12.   second=cin.get();  
  13.   
  14.   cout << "The first word began by " << first << endl;  
  15.   cout << "The second word began by " << second << endl;  
  16.   return 0;  
  17. }  

怎样得到整行数据?

如果想得到整行数据,cin有诸多限制。这时可以借用非格式化输入函数。

istream & getline(char*,int,char);

istream & getline(char* ,int);

这是istream提供的整行录入函数,但是都需要制定接收字符串的长度,streamstring提供了一个更便捷的getline版本:

istream &getline(istream&,string)

这个函数接受两个参数:一个输入流对象和一个string对象。getline函数从输入流的下一行读取,并保存读取的内容到string中,但不包括换行符。和输入操作符不一样的是,getline并不忽略行开头的换行符。只要getline遇到换行符,即便它是输入的第一个字符,getline也将停止读入并返回。如果第一个字符就是换行符,则string参数将被置为空string 

getline函数将istream参数作为返回值,和输入操作符一样也把它用作判断条件。

例如: 

[java]  view plain copy
  1. int main(){  
  2. string line;  
  3. while (getline(cin, line))   
  4.   cout << line << endl;  
  5. return 0;  
  6. }   

由于getline函数返回时丢弃换行符,换行符将不会存储在string对象中。 



参考资料:

《C++ primer》

《C++ primer plus》


你可能感兴趣的:(C语言)