-
C++ Iostreams 用法详解(二)标准输入输出
添加时间:2013-5-14 点击量:续:C++ Iostreams 用法详解(一)整体把握
起首说我们最常用的两个全局对象cin和cout,以下摘自MSDN:
You can then extract values cin or wcin to read the standard input. The rules for doing so are outlined in the description of the class basic_istream Class. You can also values to cout or wcout to write the standard output. The rules for doing so are outlined in the description of the class basic_ostream Class.
可以看出,此中cin是istream的对象,可以从标准输入中提取数据,而cout是ostream的对象,可以向标准输出中输出数据。
什么是标准输入和标准输出呢?可能对于我们这些90后来说不是那么轻易懂得的,毕竟?成果这是字符界面的概念,而我们在开端接触策画机的时辰就已经是广泛的windows图形界面的操纵体系了。然则windows操纵体系还是为我们供给了一个模仿dos这种字符界面的法度cmd,让我们大致对这个有必然的懂得。而在Linux体系中我们更轻易说清楚这个概念:在履行一个shell号令时,会默认打开三个标准文件,即标准输入文件(stdin),凡是对应终端的键盘;标准输出文件(stdout)和标准错误输出文件(stderr),这两个文件都对应终端的屏幕。过程将从标准输入文件中获得输入数据,将正常输出数据输出到标准输出文件,而将错误信息送到标准错误文件中。
如许说斗劲懂得了吧,因为我们在windows下面写一个win32 把握台应用法度的时辰,其实我们的这个法度就是类似于Linux中履行在shell中的号令,所以这个法度在履行的时辰当然可以带有参数(就是main函数的参数int argc和char argv[]了),并且有标准输入缓冲区和标准输出缓冲区的概念了。只不过我们在windows下刚开端进修编程说话的时辰,老是会用一些图形界面的IDE(例如VC++6.0就是我的发蒙IDE),所以对这个法度的运行机制并不是很清楚,只知道摁一下运行按键就主动完成了编译、连接、打开一个cmd并直接运行我们写的这个法度了。
有了标准输入和输出的概念(好把他们像Linux里面一样算作是设备文件),接下来就要说一下这个cin和这个cout到底是怎么完成输入输出机制的了。起首来看一下下面这个例子:
#include <iostream>
#include <string>
using namespace std;
int main()
{
int j = 0;
while(++j>0) for(int i=0;i>0;i++);
string str;
cin >> str;
cout << str << endl;
system(pause);
}
用嵌套的轮回来做了一个延时,的法度(win32 把握台应用法度)在正常运行的时辰,我们在键盘上敲下五个字符hello,这个时辰屏幕上并不会显示任何器材,因为固然我们的输入都保存到了输入缓冲区中,然则我们并没有把它显示到屏幕上来。第一次调用cin>>str的时辰,这时其实是调用了cin这个istream对象的>>运算符的办法,它起首让我们的过程从运行(running)状况转换到守候I/O(waiting)状况,然后将输入缓冲区中的内容都输出到屏幕上来,这个时辰我们刚才在键盘上按的hello就都显示出来了,然后持续在键盘上按五个字符world的时辰,输入缓冲区持续接收我们输入的字符,然后被cin对象显示到屏幕上来,直到我们在键盘上按回车键(正常景象下)时,这时输入缓冲区也同时接收了一个换行字符\n,这个时辰输入缓冲区中就有11个字符helloworld\n了,然后cin>>str开端将输入缓冲区中的这11个字符全部抽取(extract)出来存到对象cin的streambuf中(后面会讲到,这其实就是存储的buffer,而cin对象拥有的其实是streambuf对象的指针)。到这一步,还没有跟我们的str对象有任何干系,cin对象只是从标准输入输出中提取出了数据罢了(这时我们可以懂得为输入缓冲区中已经空了)。
之后的工作才关系到str对象,cin对象会按照str的类型(string)去格局化stream中的内容,即将前10个字符helloworld存到str对象中去,同时从streambuf中将这10个字符和停止字符\n清除(这其实是经由过程移动get指针来实现的,后面会说到),然后这条语句cin<<str;终于履行停止了。若是我们的代码改成char str[20];
cin >> str;
这时格局化工作就变为:前10个字符helloworld存到str对象中去,并将str[10]改为0作为字符串停止标记,前10个字符helloworld存到str对象中去,同时从streambuf中将这10个字符和停止字符\n “清除”。这里须要申明的是,对于每次我们应用cin>>时,起首cin断定本身的streambuf是否为空,若是是空的话,就触发一次过程状况的转换,然后开端守候用户的输入等过程。然则若是streambuf不是空的,则会先测验测验从streambuf中的数据中去格局化所需的数据。当然,这个格局化过程并不必然都是正确的,这个题目后面再论。我们在应用cin来从标准输入缓冲区中获得字符串时,有时辰应用cin >> str并不克不及解决题目,因为对于如许的操纵,在最后解析streambuf中的数据的时辰,它会以空格 、制表符\t、换行符\n来作为格局化字符串的停止字符,也就是说比如上方的代码输入hello world\n的时辰,我们获得的字符串为hello,因为在碰到空格的时辰就已经停止了字符串,重视这时streambuf中还有剩下的数据world\n,若是我们如今再次履行cin >> str则会以\n为停止字符,获取到字符串world。那若是我想一次性的获取字符串hello world要怎么做呢?istream也供给了两个获取整行字符串的(其实就是以换行符\n作为格局化字符串的停止字符)成员函数,分别就是get和getline。关于这个两个函数的应用以及其各类重载的版本等等我在这里就不久不多说了,本身去查MSDN吧。我在这里只说一下这两个函数大差别,那就是get函数在从streanbuf中格局化数据的时辰,固然也是以\n作为停止符,然则并不会清除掉streambuf中的这个字符,也就是说当我输入字符串hello world\n的时辰,get函数会正确的获得字符串hello world,然则会将字符\n留在streambuf中,而getline函数将获得与get函数一样的成果,然则不会将字符\n留在streambuf中。这有什么差别呢?差别就斗劲大了,推敲下面的代码:char str1[20],str2[20];
cin.get(str1,20);
cin.get(str2,20);
如今我输入字符串hello world\n(\n就是回车键),成功的获得了字符串str1内容为hello world,这时streambuf中还剩下\n,持续进行cin.get(str2,20),因为streambuf长短空的,就会直接格局化出str2,于是我们就获得str2的内容为,空的!然则若是我们上方应用的是getline函数,就会正常的让我们输入两个字符串了。还有,istream还供给了成员函数ignore,可以使get指针跳过字符,具体去查MSDN吧。若是在上方的代码中两个get函数之间插入cin.ignore()代码也会正常的让我们输入两个字符串。对于 cout标准输出过程,就跟上方的过程有点类似,可以从上方的例子中看到cin过程和标准输出过程是完全互不相干的。每次应用cout进行输出时,都邑主动的调用flush(将streambuf中的数据显示到屏幕上),所以在这里懂得flush并没有太大的意义,后面的文件操纵项目组也会讲到。而关于数据的格局化输出,在后面也会讲到,所以如今也就不说了。cout用起来还是斗劲简单的。关于istream和ostream的更多成员函数大师可以去查MSDN懂得。没什么写博客的经验,所以可能写的斗劲烦琐(我本身也感触感染很烦琐)。别的再夸大我还是新手,有大神发明我有说错的请必然提出啊。迎接交换,一路进修。我所有的自负皆来自我的自卑,所有的英雄气概都来自于我的软弱。嘴里振振有词是因为心里满是怀疑,深情是因为痛恨自己无情。这世界没有一件事情是虚空而生的,站在光里,背后就会有阴影,这深夜里一片寂静,是因为你还没有听见声音。—— 马良《坦白书》