对cin.get()和cin.getline()de 理解

include<iostream>
#include<cstdlib>
using namespace std;
void main()
{
 char ch;
 while(cin.get(ch))
 {
  if(ch!='#')
  
   cout<<ch;
   else
   {
    cin.putback(ch);
    break;
   }
  }
  if(!cin.eof())
  {
   cin.get(ch);
   cout<<endl<<ch<<"is next input character\n";
  }
  else
  {
   cout<<" End of file reach"<<endl;
   exit(0);
  }
  while(cin.peek()!='#')
  {
   cin.get(ch);
   cout<<ch;
  }
  if(!cin.eof())
  {
   cin.get(ch);
   cout<<endl<<ch<<"is next input character.\n";
  }
  else
  {
   cout<<" End of file reach"<<endl;
   exit(0);
  }
}
peek()函数的作用是查看下一个输入字符,假设尧都区输入,知道换行符或句点,则可以用peek()函数查看输入流中的下一个字符,以此来判断是否继续读取:
char ch;
while((ch=cin.peek()!='.'&&ch!='\n'))

现在来看下一个例子*/
#include<iostream.h>
const int SIZE=10;
inline void eatline()
{
 while(cin.get()!='\n')
  continue;
}
int main()
{
char name[SIZE];
char title[SIZE];
cout<<"enter your name"<<endl;
cin.get(name,SIZE);
if(cin.peek()!='\n')//判断是否读取了整行(也就是说,我输入rqweterytry;lkj,超过了九个,则只读到rqweteryt,也就是说没有全部读完
cout<<"sorry,we have only have enough room for" <<name<<endl;
eatline();
cout<<"dear"<<name<<" "<<"enter your title:.\n";
cin.get(title,SIZE);
if(cin.peek()!='\n')
cout<<"we only have enough room for your "<<title<<endl;
eatline();
cout<<"name"<<name<<"title"<<title<<endl;
return 0;
}

它使用peek()来确定是否读取了整行,如果一行中只有部分内容加入了输入数组中,程序将删除余下的内容


以下先解释一下cin.get()!='\n'的意思和用途
举个小程序例子
#include <iostream>
struct car
 {
 char name[20];
 int year;
};
int main(void)

{

 using namespace std;

    int n;

 cout << "How many cars do you wish to catalog?: "; 

 cin >> n;

 while(cin.get() != '\n')  // get rid of rest of line ; 
  

  car * pc = new car [n];


 int i; 

 for (i = 0; i < n; i++)

 {  
  cout << "Car #" << (i + 1) << ":\n"; 

  cout << "Please enter the make: ";

  cin.getline(pc[i].name,20);  

  cout << "Please enter the year made: ";

  cin >> pc[i].year;  

  while(cin.get() != '\n')  // get rid of rest of line  
   ;
   } 
 cout << "Here is your collection:\n"; 

 for (i = 0; i < n; i++)  

  cout << pc[i].year << " " << pc[i].name << "\n";

 delete [] pc;

 return 0;
}
使用标准控制台输入流,用户可能会有如此的输入123AJ,当输入是给一个整型时,123后面的AJ会留在输入缓冲中,
这样AJ将和后面下一次的输入一起被缓冲递交给输入处理,从而可能导致输入不合法或者出现意料外的情况,
理解这个你要明白控制台标准输入输出是被缓冲的,余下就是这条语句的理解了,其实它做的就是将后面的不合法输入在缓冲中的给清理掉,为下一次输入扫清障碍。
还要明白一点cin.get()是不会忽略空格和回车的,每次输入的最后你总是敲击enter以表明你的输入结束从而为上述语句总能处理成功提供保障。
 while(cin.get()!='\n')
  continue;
是确定第一条输入语句是否读取了整行,如果get()读取了整行,他将保留换行符,如果get()只读取了一部分
整上述代码将读取并丢弃该行中余下的内容,如果不删除余下的内容,则下一条内容将从第一个数入行中余下部分别的开始位置读取。如输入名字为
liupanminhahaha则这将导致 把ahaha读取到title中去了
cin.get()!=EOF是什么意思
就是读入的字符不是结束标志这个结束标志不是键盘上的字符,而是一个组合按键 ctrl +z ,ctrl + z连续按两次ctrl z回车,如果没退出循环,再重复一次

ch=cin,get()成员函数返回输入中的下一个字符

以下是百度搜来的以上是补充

C++的时候,这几个输入函数弄的有点迷糊;这里做个小结,为了自己复习,也希望对后来者能有所帮助,如果有差错的地方还请各位多多指教(本文所有程序均通过VC 6.0运行)

1cin
2
cin.get()
3
cin.getline()
4
getline()
5
gets()
6
getchar()

:cin.ignore();cin.get()//跳过一个字符,例如不想要的回车,空格等字符

1cin>>         

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

#include <iostream>
using namespace std;
main ()
{
int a,b;
cin>>a>>b;
cout<<a+b<<endl;
}

输入:2[回车]3[回车]
输出:5

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

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

#include <iostream>
using namespace std;
main ()
{
char a[20];
cin>>a;
cout<<a<<endl;
}

输入:jkljkljkl
输出:jkljkljkl

输入:jkljkl jkljkl       //遇空格结束
输出:jkljkl

2cin.get()

用法1 cin.get(字符变量名)可以用来接收字符

#include <iostream>
using namespace std;
main ()
{
char ch;
ch=cin.get();              //
或者
cin.get(ch);
cout<<ch<<endl;
}

输入:jljkljkl
输出:j

用法2cin.get(字符数组名,接收字符数目)用来接收一行字符串,可以接收空格

#include <iostream>
using namespace std;
main ()
{
char a[20];
cin.get(a,20);
cout<<a<<endl;
}

输入:jkl jkl jkl
输出:jkl jkl jkl

输入:abcdeabcdeabcdeabcdeabcde(输入25个字符)
输出:abcdeabcdeabcdeabcd             (接收19个字符+1'\0'

用法3cin.get(无参数)没有参数主要是用于舍弃输入流中的不需要的字符,或者舍弃回车,弥补cin.get(字符数组名,接收字符数目)的不足.

这个我还不知道怎么用,知道的前辈请赐教;

3cin.getline()  //接受一个字符串,可以接收空格并输出

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

输入:jkljkljkl
输出:jklj

接受5个字符到m中,其中最后一个为'\0',所以只看到4个字符输出;

如果把5改成20
输入:
jkljkljkl
输出:jkljkljkl

输入:jklf fjlsjf fjsdklf
输出:jklf fjlsjf fjsdklf

//延伸:
//cin.getline()
实际上有三个参数,cin.getline(接受字符串的看哦那间m,接受个数5,结束字符
)
//
当第三个参数省略时,系统默认为
'\0'
//
如果将例子中cin.getline()改为cin.getline(m,5,'a');当输入jlkjkljkl时输出jklj,输入jkaljkljkl时,输出jk

当用在多维数组中的时候,也可以用cin.getline(m[i],20)之类的用法:

#include<iostream>
#include<string>
using namespace std;

main ()
{
char m[3][20];
for(int i=0;i<3;i++)
{
cout<<"\n
请输入第"<<i+1<<"个字符串:
"<<endl;
cin.getline(m[i],20);
}

cout<<endl;
for(int j=0;j<3;j++)
cout<<"
输出m["<<j<<"]的值:"<<m[j]<<endl;

}

请输入第1个字符串:
kskr1

请输入第2个字符串:
kskr2

请输入第3个字符串:
kskr3

输出m[0]的值:kskr1
输出m[1]的值
:kskr2
输出m[2]的值:kskr3

4getline()    //接受一个字符串,可以接收空格并输出,需包含“#include<string>”

#include<iostream>
#include<string>
using namespace std;
main ()
{
string str;
getline(cin,str);
cout<<str<<endl;
}

输入:jkljkljkl
输出:jkljkljkl

输入:jkl jfksldfj jklsjfl
输出:jkl jfksldfj jklsjfl

cin.getline()类似,但是cin.getline()属于istream流,而getline()属于string流,是不一样的两个函数

注意:在使用getline时有一个奇怪的现象,vc6,要打两次回车才可以结束

5gets()       //接受一个字符串,可以接收空格并输出,需包含“#include<string>”

#include<iostream>
#include<string>
using namespace std;
main ()
{
char m[20];
gets(m);                      //
不能写成
m=gets();
cout<<m<<endl;
}

输入:jkljkljkl
输出:jkljkljkl

输入:jkl jkl jkl
输出:jkl jkl jkl

类似cin.getline()里面的一个例子,gets()同样可以用在多维数组里面:

#include<iostream>
#include<string>
using namespace std;

main ()
{
char m[3][20];
for(int i=0;i<3;i++)
{
cout<<"\n
请输入第"<<i+1<<"个字符串:
"<<endl;
gets(m[i]);
}

cout<<endl;
for(int j=0;j<3;j++)
cout<<"
输出m["<<j<<"]的值:"<<m[j]<<endl;

}

请输入第1个字符串:
kskr1

请输入第2个字符串:
kskr2

请输入第3个字符串:
kskr3

输出m[0]的值:kskr1
输出m[1]的值
:kskr2
输出m[2]的值:kskr3

自我感觉gets()cin.getline()的用法很类似,只不过cin.getline()多一个参数罢了;

里顺带说明一下,对于本文中的这个kskr1,kskr2,kskr3的例子,对于cin>>也可以适用,原因是这里输入的没有空格,如果输入了空格,比如“ks kr jkl[回车]”那么cin就会已经接收到3个字符串,“ks,kr,jkl”;再如“kskr 1[回车]kskr 2[回车]”,那么则接收“kskr,1,kskr”;这不是我们所要的结果!而cin.getline()gets()因为可以接收空格,所以不会产生这个错误;

6getchar()  //接受一个字符,需包含“#include<string>”

#include<iostream>
#include<string>
using namespace std;
main ()
{
char ch;
ch=getchar();                       //
不能写成
getchar(ch);
cout<<ch<<endl;
}

输入:jkljkljkl
输出:j

//getchar()C语言的函数,C++也可以兼容,但是尽量不用或少用;

cin.get(ch)的合法性解说

2010-10-08 14:32222人阅读评论(1)收藏举报

char ch;
cin>>ch;
while (cin.fail()==false)
{
cout<<ch;
cin>>ch;
}
cout<<endl;
//
无法识别空格

cin.get(ch);
while(cin.eof()==false){
cout<<ch;
cin.get(ch);
}

//可以识别空格

这里cin.get(ch)是把ch当参数载体放入函数,调回一个结果,所以ch应该是一个地址位置,而不是一个变量,在c语言中,这句话错了。无法调回一个结果。但是C++中支持引用当参数,所以cin.get(引用参数)可以调回一个我们需要的结果。

// cin.eof() == true ? cin.fail() == true ?(ch = cin.get()) == EOF ?

// cin.get(ch) & ch = cin.get() ->next one

// cin.get(str,size) -> end up withenter

// cin>>ch; -> ignore whitespaceand end up with enter

请问cin>>cin.get()EOF的处理机制有什么不同?

#include<iostream>
#include<iostream>
using namespace std;
int main()
{
char c;
while(!cin.eof())
cin.get(c);//cin>>c;
return 0;
}
采用cin.get()的时候运行时输入一个^Z程序就结束了。而采用cin>>的时候,输入^Z按回车程序并不结束,可以继续输入,除非紧接着又输入了一个^Z,程序才结束,太奇怪了!(VC++6.0)

原因分析如下:

输入缓冲是行缓冲。当从键盘上输入一串字符并按回车后,这些字符会首先被送到输入缓冲区中存储。每当按下回车键后,cin.get()就会检测输入缓冲区中是否有了可读的数据。cin.get()还会对键盘上是否有作为流结束标志 Ctrl+Z或者 Ctrl+D键按下作出检查,其检查的方式有两种:阻塞式以及非阻塞式。

塞式检查方式指的是只有在回车键按下之后才对此前是否有 Ctrl+Z组合键按下进行检查,非阻塞式样指的是按下 Ctrl+D之后立即响应的方式。如果在按 Ctrl+D之前已经从键盘输入了字符,则 Ctrl+D的作用就相当于回车,即把这些字符送到输入缓冲区供读取使用,此时Ctrl+D不再起流结束符的作用。如果按 Ctrl+D 之前没有任何键盘输入,则 Ctrl+D就是流结束的信号。

Windows
系统中一般采用阻塞式检查 Ctrl+ZUnix/Linux系统下一般采用非阻塞式的检查 Ctrl+D。楼主是在Windows系统下,因此使用阻塞式的 Ctrl+Z 来标识流的结束。

种阻塞式的方式有一个特点:只有按下回车之后才有可能检测在此之前是否有Ctrl+Z按下。还有一个特点就是:如果输入缓冲区中有可读的数据则不会检测Ctrl+Z(因为有要读的数据,还不能认为到了流的末尾)。还有一点需要知道:Ctrl+Z产生的不是一个普通的ASCII码值,也就是说它产生的不是一个字符,所以不会跟其它从键盘上输入的字符一样能够存放在输入缓冲区。明白了这几点之后就可以来解释楼主提出的问题了。

键盘上输入abcd^z回车之后在Windows系统上是这样处理的:由于回车的作用,前面的 abcd 等字符被送到输入缓冲区(注意:上面说过了,^z不会产生字符,所以更不会存储到输入缓冲区,缓冲区中没有 ^z 的存在)。这时,cin.get()检测到输入缓冲区中已经有数据存在(因此不再检查是否有 ^z的输入),于是从缓冲中读取相应的数据。如果都读取完了,则输入缓冲区重新变为空,cin.get()等待新的输入。可见,尽管有 ^z按下,但是由于在此之前还有其它输入字符(abcd),所以流也不会结束。

因此,输入流结束的条件就是:^z之前不能有任何字符输入(回车除外),否则 ^z起不到流结束的作用。

 

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值