Bootstrap

C++cin方法小结

一、 输入原理

 

程序的输入都建有一个缓冲区,即输入缓冲区。一次输入过程是这样的,当一次键盘输入结束时会将输入的数据存入输入缓冲区,而cin函数直接从输入缓冲区中取数据。正因为cin函数是直接从缓冲区取数据的,所以有时候当缓冲区中有残留数据时,cin函数会直接取得这些残留数据而不会请求键盘输入

#1:

#include <iostream>
using namespace std;
int main()
{
char str[8];
cin.get(str, 5);
cout<<str<<endl;
cin.get(str, 5);
cout<<str<<endl;
return 0;
}


 

测试:

abcdefgh (回车)

abcd (输出) 

efgh(输出)

【分析】之所以第一次输入完后直接程序就结束了,而不是进行第二次输入,是因为第一次多输入的数据还残留在缓存区中,第二次输入就直接从缓存区中提取而不会请求键盘输入

 

二、 几种常见的输入方式

 

(一) cin>>

该操作符是根据后面变量的类型读取数据。

输入结束条件:遇到Enter、Space、Tab键。

对结束符的处理:丢弃缓冲区中使得输入结束的结束符(Enter、Space、Tab)

cin会检查输入格式,输入与预期格式不符时,会返回false,正常情况下则会返回输入流本身;

#2:

#include <iostream>
using namespace std;
int main()
{
char str1[10], str2[10];
cin>>str1;
cin>>str2;
cout<<str1<<endl;
cout<<str2<<endl;
return 0;
}


 

测试:

abcd efgh

输出:

abcd

efgh

【分析】第一次读取字符串时遇到空格则停止了,将abcd读入str1,并舍弃了空格,将后面的字符串给了第二个字符串。这证明了cin读入数据遇到空格结束;并且丢弃空格符;缓冲区有残留数据,读入操作直接从缓冲区中取数据。

 

(二)cin.get(数组名,长度,结束符)

 

其中结束符为可选参数,读入的字符个数最多为长度减1个,结束符规定结束字符串读取的字符,默认为ENTER。

1.读取字符的情况:

若要读取字符,直接cin.get(char ch)或ch=cin.get()即可。

输入结束条件:Enter键

对结束符处理:不丢弃缓冲区中的Enter

cin.get()与cin.get(char ch)用于读取字符,他们的使用是相似的,即:ch=cin.get()与cin.get(ch)是等价的。若文件读取最好使用文件流.get(ch)。

#3:

#include <iostream>
using namespace std;
int main()
{
   char c1, c2;
   cin.get(c1);
   cin.get(c2);
   cout<<c1<<“”<<c2<<endl; // 打印两个字符
   cout<<(int)c1<<“”<<(int)c2<<endl; // 打印这两个字符的ASCII值
   return 0;
 }


测试一输入:

a[Enter]

输出:

a

 

97 10

测试二输入:

a b[Enter]

输出:

a

97 32

 

【测试一分析】会发现只执行了一次从键盘输入,显然第一个字符变量取的’a', 第二个变量取的是Enter(ASCII值为10),这是因为该函数不丢弃上次输入结束时的Enter字符,所以第一次输入结束时缓冲区中残留的是上次输入结束时的Enter字符!

【测试二分析】显然第一个字符变量取的’a', 第二个变量取的是Space(ASCII值为32)。原因同上,没有丢弃Space字符。

 

2.读取字符串的情况:

输入结束条件:默认Enter键(因此可接受空格,Tab键),可在第三个参数上自定义结束符

对结束符处理:默认时丢弃缓冲区中的Enter,自定义结束符时不丢弃缓冲区中的结束符

#4:

#include <iostream>
using namespace std;
int main ()
{
  char ch, a[20];
  cin.get(a, 5 , 'd');
  cin>>ch;
  cout<<a<<endl;
  cout<<(int)ch<<endl;
  return 0;
}


测试一输入:

12345[Enter]

输出:

1234

53

测试二输入:

12d45[Enter]

输出:

12

d

 

【测试一分析】第一次输入超长,字符串按长度取了"1234",而’5′仍残留在缓冲区中,所以第二次输入字符没有从键盘读。

入,而是直接取了’5′,所以打印的ASCII值是53(’5′的ASCII值)。

【测试二分析】第二次输出为d,说明自定义结束符时不丢弃缓冲区中的结束符。

 

(三)cin.getline()

 

cin.getline(数组名,长度,结束符) 大体与 cin.get(数组名,长度,结束符)类似。(实际能读取长度(-1)个字符,因为字符串结尾是0);

区别在于:

cin.get()当输入的字符串超长时,不会引起cin函数的错误,后面的cin操作会继续执行,只是直接从缓冲区中取数据。但是cin.getline()当输入超长时,会引起cin函数的错误,后面的cin操作将不再执行。

#5: 

#include <iostream>
using namespace std;
int main ()
{
  char ch, a[20];
  cin.getline(a, 5);
  cin>>ch;
  cout<<a<<endl;
  cout<<(int)ch<<endl;
  return 0;
}


测试输入:

12345[Enter]

输出:

1234

-52

【分析】与cin.get()的例子比较会发现,这里的ch并没有读取缓冲区中的5,而是返回了-52,这里其实cin>>ch语句没有执行,是因为cin出错了!

四 cin.read(c, n)

从字符串流中读取n个字符到c数组中。

char score[20]; 

cin.read(score,20); 

 

三、 其它方法

(一)cin.putback(x)

将x内容置入缓冲区。

#1

#include<iostream>
 
using namespace std;
 
int main(){
char a = ‘m’,b;     
cin.putback(a);
cin>>b; 
cout<<a<<endl;
cout<<b;
}


测试输入:无输入

输出:

m

m

【分析】将a中数据置入缓冲区后,直接流入b中。

(二)cin.peek()

返回缓冲区中的下一个字符,但只是查看,并不从缓冲区中取出。

#2

#include<iostream>
 
using namespace std;
 int main(){
char input[100]; 
char ch; 
int i=0; 
while((ch=cin.peek())!='.'&&ch!='\n') 
    cin.get(input[i++]); 
input[i]='\0'; 
}


【分析】程序遇到句号或换行符循环停止。句点或换行符仍停留在输入流中。可见,使用peek的效果相当于先用get()读取一个字符,再用putback()将字符放入输入流中。

 

(三)cin.ignore(长度,结束符)

cin.ignore(a,ch)方法是从输入流(cin)中提取字符,提取的字符被忽略(ignore),不被使用。每抛弃一个字符,它都要计数和比较字符:如果计数值达到a或者被抛弃的字符是ch,则cin.ignore()函数执行终止;否则,它继续等待。它的一个常用功能就是用来清除以回车结束的输入缓冲区的内容,消除上一次输入对下一次输入的影响。比如可以这么用:cin.ignore(1024,'\n'),通常把第一个参数设置得足够大,这样实际上总是只有第二个参数'\n'起作用,所以这一句就是把回车(包括回车)之前的所以字符从输入缓冲(流)中清除出去。

#3

#include <iostream>
using namespace std;
int main()
{
       char array[8];
       cin.ignore(6,‘a’);       //吸收6个字符或读取到a使array无法读取;
       cin.getline(array,8);
       cout<<array<<endl;
       return 0;
}


 

(四)in.clear()、cin.sync()

cin.clear()是用来更改cin的状态标示符的。

cin.sync()是用来清除缓存区的数据流的。
如果标示符没有改变那么即使清除了数据流也无法输入。所以两个要联合起来使用。

例如:

#4

#include<iostream>
using namespace std;
int main(){
	int a;cout<<"输入一个字母:"<<endl;
	cin>>a;//int型变量中放了char型数据,failbit置1
	cout<<"cin.fail()="<<cin.fail()<<endl;//输出1
	//cin.clear();//cin.sync();
	cout<<"输入一个数字:"<<endl;//由于failbit值为1,输入流不能正常工作
	cin>>a;//故此处的输入无效
	cout<<a<<endl;//输出不确定值
	cin.clear();//此处用cin.clear()流标志复位
	//cin.sync();
	cout<<"cin.fail()="<<cin.fail()<<endl;//此处failbit已为0
	cout<<"输入一个数字:"<<endl;//但刚才输入的字符并没有从流中清除,所以cin>>a又把那个字符放入a中,流输入流又不能正常工作
	cin>>a;cout<<a<<endl; //输出不确定值
	cout<<"cin.fail()="<<cin.fail()<<endl;//在此处failbit又为1
	cin.clear();//再次修复输入流
	cin.sync();//取走刚才流中的字符
	cout<<"输入一个数字:"<<endl;//再次接收用记输入,这次输入数字,正常输出了
	cin>>a;
	cout<<"a="<<a<<endl;//现在再看一下输入流的failbit
	cout<<"cin.fail()="<<cin.fail()<<endl;//输出0,表明输入流已恢复正常
	return 0;
}


 

(五)in.width(长度)

接收长度-1个字符,其他的放在流中等待接收

#5

#include<iostream>
using namespace std;
int main()
{
       int w=4;
       char str[10];
       cout<<"Entera sentence:\n";
       cin.width(5); //每次只接收4个字符,其他的放在流中等待接收
       while( cin>> str)
       {
              cout.width(w++);//将4个字符输出,设置每次输出的域宽增加1
             cout<<str<<endl; //输出字符
             cin.width(5);    //设置接收4个字符
       }
       return 0;
}


输入:

happy new year

输出:

happ

    y

   new

   year

^Z

【分析】app(四个字符),y留在下一次。遇到空格接收结束,第二次只有y,到y输出时,输出域宽是5。下一个接收的是new(后面的空格断开了接收),……

 

(六) 数据进行合法性检查

 

cin对象有几个专门用来报告其工作情况的成员函数,他们将返回一个真/假值来表明cin的状态。

-eof():如果到达文件(或输入)末尾,返回true;

-fail():如果cin无法工作,返回ture;

-bad():如果cin因为比较严重的原因(例如内存不足)而无法工作,返回true;

-good():如果以上情况都没发生,返回true;


;