【听说有人想转码】输入之“除了scanf还有啥”--cin与cin.get与cin.getline

cin、cin.get()、cin.getline()、getline()的区别_a3192048的博客-CSDN博客_cin.getline

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/a3192048/article/details/80303547
————————————————
版权声明:本文为CSDN博主「啊大1号」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/a3192048/article/details/80303547

本人看了c站博主啊大1号的文章,受益匪浅,在此记录拜读经历,侵权删。(点击上方的链接可看原文)

为了能做笔记,我把他的文章复制粘贴至word。黄色底是概念,红色底是代码书写格式。标出来供以后查阅。(怕以后找不到这些内容,特意写了篇文章供以后寻物~)

格式 概念

测试平台g++ 5.4.0和VS2015社区版。

输入原理简述:

程序的输入都建有一个缓冲区,即输入缓冲区。每次输入过程是这样的,当一次键盘输入结束时会将输入的数据存入输入缓冲区,而cin函数直接从输入缓冲区中取数据(cin读取数据是从第一个非空白字符开始到下一个空白字符结束)。正因为cin函数是直接从缓冲区取数据的,所以有时候当缓冲区中有残留数据时,cin函数会直接取得这些残留数据而不会请求键盘输入。

  比如下面的例子:

void test_input()  

{  

    string str;  

    cout<<"cin的测试:"<<endl;  

    cin>>str;  

    cout<<str<<endl;  

    cin>>str;  

    cout<<str<<endl;  

}  

        由于cin在遇到空格/tab时,就会停止读取,所以如果我在第一次输入时,利用空格隔开两个字符串,那么cin在第一次取的时候,只会读取前一个字符串,到空格结束,此时缓冲区还保留着前面输入的第二个字符串,那么第二次cin就会直接从缓冲区取残留数据,而不会请求输入。

       当然对于以上的情况,也有解决的方案,那就是在第二次调用cin>>str之前通过cin.sync()来清空输入缓冲区,看一下下面的例子,此处不赘述:

void test_input()  

{  

    string str;  

    cout<<"cin的测试:"<<endl;  

    cin>>str;  

    cin.sync();  

    cout<<str<<endl;  

    cin>>str;  

    cout<<str<<endl;  

}  

 各种输入方法简介:

        1、cin>>

        根据cin>>sth 中sth的变量类型读取数据,这里变量类型可以为int,float,char,char*,string等诸多类型。这一输入操作,在遇到结束符(Space、Tab、Enter)就结束,且对于结束符,并不保存到变量中。注意:最后一个enter也在缓冲区。

void test_input()  

{  

    char ch;

    char ch1[10],ch2[10];  

    cout<<"输入两个字符串:"<<endl;  

    cin>>ch1;  

    cin>>ch2;  

    cout<<"两个字符串分别为:"<<endl;  

    cout<<ch1<<endl;  

    cout<<ch2<<endl;  

    cin.get(ch);

    cout << 强制转化为整形,只取整形部分(int)ch << endl; //输出10,为最后一个enter输入

}  

2、cin.get(字符数组名,接收长度,结束符)

其中结束符意味着遇到该符号结束字符串读取,默认为enter,读取的字符个数最多为(长度 - 1),因为最后一个为'\0'。要注意的是,cin.get(字符数组名,接收长度,结束符)操作遇到结束符停止读取,但并不会将结束符从缓冲区丢弃。cin.get函数有如下几种声明:

int get();

istream& get(char& c);

istream& get(char* s, streamsize n);

istream& get(char* s, streamsize n, char delim);

istream& get(streambuf& sb);

istream& get(streambuf& sb, char delim);

(1)接收一个字符ch=cin.get()或cin.get(char ch),二者等价,看两个例子

cin.get()存在的基本目的,我认为就是为了从c移植到c++的时候,直接用cin.get()代替getchar(),也正因因此,cin.get()的返回值跟其它cin.get成员函数返回cin对象不同,跟getchar()一样返回int。所以cin.get()和C语言的getchar()没什么区别。

void test_input()  

{  

    char ch1,ch2;  

    cout<<"请输入两个字符:"<<endl;  

    cin.get(ch1);//或ch1 = cin.get();  

    cin.get(ch2);  

    cout<<ch1<<" "<<ch2<<endl;  

    cout<<(int)ch1<<" "<<(int)ch2<<endl;  

}  

来看几组测试:

连续输入ab[enter],结果正常,ch1,ch2分别读取了a、b,将其输出,然后在输出其ASCII值。要注意的是,以上输入并读取后,缓冲区中依然存在[Enter]没有被删除。

输入a[Space]b[Enter],结果在输出时,只看到了a,输出ASCII值时候分别为97 32(空格的ASCII值),这就说明cin.get()并不会舍弃Space,依然会将其读取进去,并加以显示等操作。

输入a[Enter],输出见下图。在输出a之后,第二次的输出产生了换行的效果,而输出的第二个ASCII值为10(Enter的ASCII值),这就进一步响应了前面说到的cin.get()遇到结束符并不会将之删除。  

(2)接收一定长度的字符串cin.get(字符数组名,接收长度,结束符),结束符为可选参数,默认为Enter,可以接受Space 、Tab,对于结束符的处理要注意,结束符并不会丢掉,同样看几个例子。

对于如下代码,所做的操作时,在不遇到enter时最多读入(6-1)=5个字符到ch2中,然后读入下一个字符到ch1中,显示ch2,ch1以及其ASCII码值。附上几组测试:  

void test_input()  

{  

    char ch1,ch2[10];  

    cout<<"请输入字符串:"<<endl;  

    cin.get(ch2,6);//在不遇到结束符的情况下,最多可接收6-1=5个字符到ch2中,注意结束符为默认Enter  

    cin.get(ch1);//或ch1 = cin.get();  

    cout<<ch2<<endl;  

    cout<<ch1<<"\n"<<(int)ch1<<endl;  

}  

输入:zifuchuan[Enter],由于输入长度大于(6-1)=5,所以会首先读入“zifuc”到ch1,此时“huan”仍在缓冲区,当执行cin.get(ch1)会直接从缓冲区读入h,而不需要申请从键盘输入,看一下结果,符合分析。

输入:zifu[Enter],此时输入长度小于5就遇到了默认结束符Enter,则ch2中只读入“zifu”,要注意的是,输入缓冲区里面的Enter还在,所以接下来要读入的ch1的内容将是Enter,而输出时将看到换行,ASCII码值为10,见下图

输入:zi fuchuan[Enter],注意中间的空格,cin.get()对空格并不敏感,依然会读入,故而ch2读入的是“zi fu”,ch1读入的是c

假如:

#include <iostream>

using namespace std;

int main()

{

char ch1, ch2[10];

cout << "请输入字符串:" << endl;

cin.get(ch2, 13);//在不遇到结束符的情况下,最多可接收13-1=12个字符到ch2中,注意结束符为默认Enter  

cin.get(ch1);//或ch1 = cin.get();  

cout << ch2 << endl;

cout << ch1 << "\n" << (int)ch1 << endl;

return 0;

}

输入:123456789(enter)

输出:

输入:1234567890(enter)

则VS2015运行报错!因为ch2[10]用cin.get最多接受9个字符,最后一个是'\0'。但g++ 5.4.0正常(说明没有对下标越界做检查)。

还有一点需要注意的是,cin.get(字符数组名,接收长度,结束符),当一开始第一个输入字符(即前面无其他任何字符)就遇到结束符情况下,将不会正常输出,但缓冲区中依然有该结束符。

例:

#include <iostream>

using namespace std;

int main()

{

char ch[5], ch2;

cin.get(ch, 3); //默认结束符'\n',此处直接(enter)

cin.get(ch2);

cout << ch2 << ' ' << (int)ch2 << endl; //VS2015 输出-52, g++ 5.4.0 输出0

return 0;

}

以上程序如果输入1(enter),将正常输出\n的ASCII码,输出如下:

清除错误标志:

#include <iostream>

using namespace std;

int main()

{

char ch[5], ch2;

cin.get(ch, 3); //默认结束符'\n',此处直接(enter)

cin.clear(); //清除错误标志

cin.get(ch2);

cout << ch2 << ' ' << (int)ch2 << endl;

return 0;

}

输出:

可见结束符依然在缓冲区当中。

再来一例:

#include <iostream>

using namespace std;

int main()

{

char ch[5], ch2;

cin.get(ch, 3, 'a');// 结束符为'a',直接输入a(enter)

cin.get(ch2);

cout << ch2 << ' ' << (int)ch2 << endl;

return 0;

}

输出:

如果输入1a(enter),输出如下:

清除错误标志:

#include <iostream>

using namespace std;

int main()

{

char ch[5], ch2;

cin.get(ch, 3, 'a');// 结束符为'a',直接输入a(enter)

cin.clear(); // 清除错误标志

cin.get(ch2);

cout << ch2 << ' ' << (int)ch2 << endl;

return 0;

}

输出:

可见结束符'a'依然在缓冲区当中。

补充:cin.get(ch2)和ch2 = cin.get()小区别:

#include <iostream>

using namespace std;

int main()

{

char ch[5], ch2;

cin.get(ch, 3, 'a'); //此处输入a(enter)

ch2 = cin.get(); //注意与cin.get(ch2)不同

cout << ch2 << ' ' << (int)ch2 << endl;

return 0;

}

(3)cin.get( ),注意此时没有参数,可用于舍弃输入流中的不需要的字符,或者舍弃回车,弥补cin.get(字符数组名,字符数目,结束符)的不足。对(2)中的代码加入一句话cin.get()如下:

void test_input()  

{  

    char ch1,ch2[10];  

    cout<<"请输入字符串:"<<endl;  

    cin.get(ch2,6);//在不遇到结束符的情况下,最多可接收6-1=5个字符到ch2中  

    cin.get();//注意:前面两句可以写到一块:cin.get(ch2,6).get();  

    cin.get(ch1);//或ch1 = cin.get();  

    cout<<ch2<<endl;  

    cout<<ch1<<"\n"<<(int)ch1<<endl;  

}  

前面遇到的一个状况是,输入字符后,其结束符(如默认的Enter)会保留在缓冲区中,当下次读入时,又会再读入,此时就可以用到cin.get()读掉输入缓冲区不需要的字符,如:输入:zi[Enter],由于遇到结束符,所以ch2内容为zi,此时输入缓冲区还存在着[Enter],但cin.get()将其舍弃掉之后,cin.get(ch1)就会申请从键盘输入内容,如下所示:

3、cin.getline(字符数组名,接收长度,结束符)

其用法与cin.get(字符数组名,接收长度,结束符)极为类似。cin.get()当输入的字符串超长时,不会引起cin函数的错误,后面若有cin操作,会继续执行,只是直接从缓冲区中取数据。但是cin.getline()当输入超长时,会引起cin函数的错误,后面的cin操作将不再执行。如下代码:

void test_input()  

{  

    char ch1,ch2[10];  

    cout<<"请输入字符串:"<<endl;  

    cin.getline(ch2,6);//在不遇到结束符的情况下,最多可接收6-1=5个字符到ch2中  

    cin>>ch1;  

    cout<<ch2<<endl;  

    cout<<ch1<<"\n"<<(int)ch1<<endl;  

}  

 测试:如下图,输入zifuchuan[Enter],长度大于最大长度5,就会导致cin函数错误,其后既没有像cin.get()一样直接从输入缓冲区直接读数据,也没有从键盘输入。所以此处可以注意,考虑在用cin.getline()时,适度设置其最大接受长度大一点。

假如:

#include <iostream>

using namespace std;

int main()

{

char ch1, ch2[6]; // 最多可接收5个,最后一个'\0'

cout << "请输入字符串:" << endl;

cin.getline(ch2, 8);//在不遇到结束符的情况下,最多可接收8-1=7个字符到ch2中  

cin >> ch1;

cout << ch2 << endl;

cout << ch1 << "\n" << (int)ch1 << endl;

return 0;

}

输入:123456(enter)     VS将会报错,g++正常。说明VS对数组下标进行了检查。

cin.get()每次读取一整行并把由Enter键生成的换行符留在输入队列中,然而cin.getline()每次读取一整行并把由Enter键生成的换行符抛弃,比如:

#include <iostream>

using namespace std;

int main() {

cout << "Enter your name:";

char name[15];

//cin.get(name, 15);

cin.getline(name, 15);// 输入abc(enter)

cout << "name:" << name << endl;

char ch;

cin.get(ch);// 输入123(enter) 注:因为cin.getline把最后一个换行符丢弃了,所以此处ch读取字符'1'

cout << (int)ch << endl;  //输出49  '1'的ASCII码值

cout << "\nEnter your address:";

char address[15];

//cin.getline(address, 15);

cin.get(address, 15); //直接读取缓冲区剩余的23 注:cin.get保留最后一个换行符在缓冲区

cout << "address:" << address << endl;

cin.get(ch); // 读取缓冲区最后一个换行符

cout << (int)ch << endl;

cin.get(ch);  // 缓冲区已为空,所以从键盘输入a

cout << (int)ch << endl; //输出97  'a'的ASCII码值

return 0;

}

#include <iostream>

using namespace std;

int main() {

cout << "Enter your name:";

char name[15];

cin.get(name, 15); // 输入abc(enter)

//cin.getline(name, 15);

cout << "name:" << name << endl;

char ch;

cin.get(ch); //因为cin.get不会丢弃最后一个换行符,所以此处ch读取换行符

cout << (int)ch << endl; //输出10  '\n'的ASCII码值

cout << "\nEnter your address:";

char address[15];

cin.getline(address, 15); //输入123(enter)

//cin.get(address, 15);

cout << "address:" << address << endl;

cin.get(ch);// 因为cin.getline丢弃最后一个换行符,所以此处重新从键盘输入a(enter)

cout << (int)ch << endl; //输出97 'a'的ASCII码值

cin.get(ch);   // cin.get不会丢弃最后一个换行符,所以此处读取上一步输入的保留在缓冲区的换行符

cout << (int)ch << endl; // 输出10 '\n'的ASCII码值

return 0;

}

4、getline(istream is,string str,结束符)

同样,此处结束符为可选参数(默认依然为enter)。然而,getline()与前面的诸多存在的差别在于,它string库函数下,而非前面的istream流,所有调用前要在前面加入#include<string>。与之对应这一方法读入时第二个参数为string类型,而不再是char*,要注意区别。另外,该方法也不是遇到空白字符(tab, space, enter(当结束符不是默认enter时))就结束输入的,且会丢弃最后一个换行符。

#include <iostream>

#include <string>

using namespace std;

int main() {

char ch;

string str;

cout << "请输入string内容:" << endl;

getline(cin, str); //输入:abc(space)(space)d(enter)

//getline(cin, str, 'a');

cout << str << endl;

cin.get(ch); //因为丢弃了最后一个换行符,所以此处从键盘输入b

cout << (int)ch << endl; //输出:98 'b'的ASCII码值

return 0;

}

#include <iostream>

#include <string>

using namespace std;

int main() {

char ch;

string str;

cout << "请输入string内容:" << endl;

//getline(cin, str);

getline(cin, str, 'a'); //输入:string(enter空了一行)strinabc,遇到结束符'a'停止读取

cout << str << endl; //输出string(enter)strin

cin.get(ch); //将缓冲区中'a'丢弃,读取'a'之后的'b'

cout << (int)ch << endl; //输出:98 'b'的ASCII码值

return 0;

}

5、getline和cin.getline区别

cin.getline()当输入超长时,会引起cin函数的错误,后面的cin操作将不再执行。

#include <iostream>  

using namespace std;

int main()

{

char ch, a[20];

cin.getline(a, 5);// 此处输入12345(enter)

cin >> ch;

cout << a << endl;

cout << (int)ch << endl;

return 0;

}

输出:

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值