C++中关于输入cin、gets、getline等的一些整理

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/wuzhongqiang/article/details/77587732

C++中的输入有很多,它既可以兼容C语言中的一些输入,像scanf,gets,getchar()。同时也有自己的一些输入方法,像cin,cin.get,getline,ges,等,下面就分享一下c++输入方法的一些区别:


输入原理简述:
程序的输入都建有一个缓冲区,即输入缓冲区。每次输入过程是这样的,当一次键盘输入结束时会将输入的数据存入输入缓冲区,而cin函数直接从输入缓冲区中取数据。正因为cin函数是直接从缓冲区取数据的,所以 有时候当缓冲区中有残留数据时,cin函数会直接取得这些残留数据而不会请求键盘输入。
比如下面的例子:
1
2
3
4
5
6
7
8
9
void test_input()
{
string str;
cout<<"cin的测试:"<<endl;
cin>>str;
cout<<str<<endl;
cin>>str;
cout<<str<<endl;
}

由于cin在遇到空格时,就会停止输入,所以如果我在第一次输入时,利用空格隔开两个字符串,那么cin在第一次取的时候,只会读取前一个字符串,到空格结束,此时缓冲区还保留着前面输入的第二个字符串,那么第二次cin就会直接从缓冲区取残留数据,而不会请求输入。
当然对于以上的情况,也有解决的方案,那就是在第二次调用cin>>str,之前通过cin.sync()来清空输入缓冲区,看一下下面的例子,此处不赘述:
1
2
3
4
5
6
7
8
9
10
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)就结束,且对于结束符,并不保存到sth中。
1
2
3
4
5
6
7
8
9
10
void test_input()
{
char ch1[10],ch2[10];
cout<<"输入两个字符串:"<<endl;
cin>>ch1;
cin>>ch2;
cout<<"两个字符串分别为:"<<endl;
cout<<ch1<<endl;
cout<<ch2<<endl;
}
2、cin.get(字符数组名,接收长度,结束符)
其中结束符意味着遇到该符号结束字符串读取,默认为ENTER,读取的字符个数最多为(长度-1),因为最后一个为”n”。要注意的是,cin.get()操作遇到结束符停止读取, 但并不会将结束符从缓冲区丢弃 。cin.get()有如下几种用法:
(1)接收一个字符ch=cin.get()或cin.get(char ch),二者等价,看两个例子
1
2
3
4
5
6
7
8
9
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码值。附上几组测试:
1
2
3
4
5
6
7
8
9
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
(3)cin.get( ),注意此时没有参数可用于舍弃输入流中的不需要的字符,或者舍弃回车,弥补cin.get(字符数组名,字符数目,结束符)的不足。对(2)中的代码加入一句话cin.get()如下:
1
2
3
4
5
6
7
8
9
10
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操作将不再执行。如下代码:
1
2
3
4
5
6
7
8
9
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()时,适度设置其最大接受长度大一点。
4、getline(istream is,string str,结束符)
同样,此处结束符为可选参数(默认依然为Enter)。然而,getline()与前面的诸多存在的差别在于,它string库函数下,而非前面的istream流,所有调用前要在前面加入#include。 与之对应这一方法读入时第二个参数为string类型,而不再是char* ,要注意区别。另外, 该方法也不是遇到空格就结束输入的。
1
2
3
4
5
6
7
void test_input()
{
string str;
cout<<"请输入string内容:"<<endl;
getline(cin,str,'a');
cout<<str<<endl;
}
通过以上第二个图还可以看出, 这一方法只有在遇到结束符(此处为‘a’)才结束,对空格甚至回车都不敏感。

5、gets(char *ch)
gets()方法同样接受一个字符串,但是与getline()不同, 它的参数为char*, 而不是string,另外若定义char ch[n],长度为n,则样注意输入的字符串长度不要大于n,否则会报错。 同样gets()对空格也不敏感。
1
2
3
4
5
6
7
void test_input()
{
char ch[10];
cout<<"请输入char*内容:"<<endl;
gets(ch);
cout<<ch<<endl;
}
ee3-tmp

输入原理简述:
程序的输入都建有一个缓冲区,即输入缓冲区。每次输入过程是这样的,当一次键盘输入结束时会将输入的数据存入输入缓冲区,而cin函数直接从输入缓冲区中取数据。正因为cin函数是直接从缓冲区取数据的,所以 有时候当缓冲区中有残留数据时,cin函数会直接取得这些残留数据而不会请求键盘输入。
比如下面的例子:
1
2
3
4
5
6
7
8
9
void test_input()
{
string str;
cout<<"cin的测试:"<<endl;
cin>>str;
cout<<str<<endl;
cin>>str;
cout<<str<<endl;
}

由于cin在遇到空格时,就会停止输入,所以如果我在第一次输入时,利用空格隔开两个字符串,那么cin在第一次取的时候,只会读取前一个字符串,到空格结束,此时缓冲区还保留着前面输入的第二个字符串,那么第二次cin就会直接从缓冲区取残留数据,而不会请求输入。
当然对于以上的情况,也有解决的方案,那就是在第二次调用cin>>str,之前通过cin.sync()来清空输入缓冲区,看一下下面的例子,此处不赘述:
1
2
3
4
5
6
7
8
9
10
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)就结束,且对于结束符,并不保存到sth中。
1
2
3
4
5
6
7
8
9
10
void test_input()
{
char ch1[10],ch2[10];
cout<<"输入两个字符串:"<<endl;
cin>>ch1;
cin>>ch2;
cout<<"两个字符串分别为:"<<endl;
cout<<ch1<<endl;
cout<<ch2<<endl;
}
2、cin.get(字符数组名,接收长度,结束符)
其中结束符意味着遇到该符号结束字符串读取,默认为ENTER,读取的字符个数最多为(长度-1),因为最后一个为”n”。要注意的是,cin.get()操作遇到结束符停止读取, 但并不会将结束符从缓冲区丢弃 。cin.get()有如下几种用法:
(1)接收一个字符ch=cin.get()或cin.get(char ch),二者等价,看两个例子
1
2
3
4
5
6
7
8
9
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码值。附上几组测试:
1
2
3
4
5
6
7
8
9
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
(3)cin.get( ),注意此时没有参数可用于舍弃输入流中的不需要的字符,或者舍弃回车,弥补cin.get(字符数组名,字符数目,结束符)的不足。对(2)中的代码加入一句话cin.get()如下:
1
2
3
4
5
6
7
8
9
10
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操作将不再执行。如下代码:
1
2
3
4
5
6
7
8
9
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()时,适度设置其最大接受长度大一点。
4、getline(istream is,string str,结束符)
同样,此处结束符为可选参数(默认依然为Enter)。然而,getline()与前面的诸多存在的差别在于,它string库函数下,而非前面的istream流,所有调用前要在前面加入#include。 与之对应这一方法读入时第二个参数为string类型,而不再是char* ,要注意区别。另外, 该方法也不是遇到空格就结束输入的。
1
2
3
4
5
6
7
void test_input()
{
string str;
cout<<"请输入string内容:"<<endl;
getline(cin,str,'a');
cout<<str<<endl;
}
通过以上第二个图还可以看出, 这一方法只有在遇到结束符(此处为‘a’)才结束,对空格甚至回车都不敏感。

5、gets(char *ch)
gets()方法同样接受一个字符串,但是与getline()不同, 它的参数为char*, 而不是string,另外若定义char ch[n],长度为n,则样注意输入的字符串长度不要大于n,否则会报错。 同样gets()对空格也不敏感。
1
2
3
4
5
6
7
void test_input()
{
char ch[10];
cout<<"请输入char*内容:"<<endl;
gets(ch);
cout<<ch<<endl;
}

6、scanf()
scanf(), getchar()等都是标准输入函数,下面通过一段简单的程序看一下区别:
  #include <stdio.h>
    int main()
    {
char ch1, ch2;
scanf("%c", &ch1); 
scanf("%c", &ch2);
printf("%d %d\n", ch1, ch2);
return 0;
    }
    或者是:
    #include <stdio.h>
    int main()
    {
char ch1, ch2;
ch1 = getchar();
ch2 = getchar();
printf("%d %d\n", ch1, ch2);
return 0;
    }
    程序的本意很简单,就是从键盘读入两个字符,然后打印出这两个字符的ASCII码值。可是执行程序后会发现除了问题:当从键盘输入一个字符后,就打印出了结果,根本就没有输入第二个字符程序就结束了。例如用户输入字符'a', 打印结果是97,10。这是为什么呢?
【分析】:
    首先我们呢看一下输入操作的原理, 程序的输入都建有一个缓冲区,即输入缓冲区。一次输入过程是这样的,当一次键盘输入结束时会将输入的数据存入输入缓冲区,而 scanf函数直接从输入缓冲区中取数据。正因为scanf函数是直接从缓冲区取数据的,所以有时候当缓冲区中有残留数据时,scanf函数会直接取得这些残留数据而不会请求键盘输入 ,这就是例子中为什么会出现输入语句失效的原因!
    其实这里的10恰好是回车符! 这是因为scanf()和getchar()函数是从输入流缓冲区中读取值的,而并非从键盘(也就是终端)缓冲区读取。 而读取时遇到回车(\n)而结束的,这个\n会一起读入输入流缓冲区的,所以第一次接受输入时取走字符后会留下字符\n,这样第二次的读入函数直接从缓冲区中把\n取走了,显然读取成功了,所以不会再从终端读取!这就是为什么这个程序只执行了一次输入操作就结束的原因!

scanf()读取字符串时:
#include <stdio.h>
    int main()
    {
char str1[20], str2[20];
scanf("%s",str1); 
printf("%s\n",str1);    
scanf("%s",str2); 
printf("%s\n",str2); 
return 0;
    }
    程序的功能是读入一个字符串输出,在读入一个字符串输出。可我们会发现输入的字符串中不能出现空格,例如:
测试 一输入:
Hello world!
输出:
Hello
world!
【分析】到此程序执行完毕,不会执行第二次的读取操作!这个问题的原因跟问题一类似,第一次输入Hello world!后,字符串Hello world!都会被读到输入缓冲区中, 而scanf()函数取数据是遇到回车、空格、TAB就会停止,也就是第一个scanf()会取出"Hello", 而"world!"还在缓冲区中,这样第二个scanf会直接取出这些数据,而不会等待从终端输入。
测试二:
Hello[Enter] 
Hello[输出]
world[Enter]
world[输出]
【分析】程序执行了两次从键盘读入字符串,说明第一次输入结束时的回车符被丢弃!即:scanf()读取字符串会舍弃最后的回车符!

我们再看一下gets()读取字符串的情况:
用scanf来读取一个字符串时,字符串中是不可以出现空格的,一旦出现空格,后面的数据就会舍弃残留在缓冲区中。 其实有另外一个函数是可以接受空格的,那就是gets(),下面我们看一下这个函数的应用,我们把程序2改动一下:
程序3:
#include <stdio.h>
int main()
{
char str1[20], str2[20];
gets(str1); 
printf("%s\n",str1);    
gets(str2); 
printf("%s\n",str2); 
return 0;
}
测试:
Hello world! [输入]
Hello world! [输出]
12345 [输入]
12345 [输出]
【分析】显然与上一个程序的执行情况不同,这次程序执行了两次从键盘的读入,而且第一个字符串取了Hello world! 接受了空格符,而没有像上一个程序那样分成了两个字符串!所以如果要读入一个带空格符的字符串时因该用gets(), 而不宜用scanf()!

scanf、getchar、gets的总结:
第一:要注意不同的函数是否接受空格符、是否舍弃最后的回车符的问题!
读取字符时:
scanf()以Space、Enter、Tab结束一次输入,不会舍弃最后的回车符(即回车符会残留在缓冲区中);
getchar()以Enter结束输入,也不会舍弃最后的回车符;
读取字符串时:
scanf()以Space、Enter、Tab结束一次输入
gets()以Enter结束输入(空格不结束),接受空格,会舍弃最后的回车符!

当然,这些输入方法肯定不会都那么常用,在这简述一下区别之后以后用的时候就有了了解,有的题目可能考这个,在PAT乙级题中好几个题考到了这个,所以要写程序的时候要考虑好输入,毕竟先有输入,后有输出,一旦输入出了问题,程序写的再好,也是白搭。
Pat乙级中常用的输入scanf cin getline getchar() gets等。

ee3-tmp

猜你喜欢

转载自blog.csdn.net/wuzhongqiang/article/details/77587732