[C++]C++中的IO类

C++中的IO类

C++语言不直接处理输入输出,而是通过一组定义在标准库中的类型来处理IO。这些类型支持从设备读取数据,向设备写入数据的IO操作,设备可以是文件,控制台窗口等。还有一些类型允许内存IO,即从string读取数据,向string写入数据等。

应用程序不只从控制台窗口进行IO操作,常常需要读写命名文件,并且使用IO操作处理string的字符会很方便,所以在istream和ostream之外,标准库还定义了其他的一些IO类型,分别定义在三个独立的头文件中:iostream定义了用于读写流的基本类型,fstream定义了读写命名文件的类型,sstream定义了读写内存string对象的类型。

类型ifstream和istringstream都继承自istream。因此我们可以像使用istream对象一样来使用ifstream和istringstream对象。需要注意的是,不能对IO对象进行拷贝或赋值,另外读写一个IO对象会改变其状态,因此传递和返回的引用不能是const的

条件状态

IO类定义了一些函数和标志,可以帮助我们访问和操纵流的条件状态。一个流一旦发生错误,其后的IO操作都会失败,确定一个流对象的最简单的方法就是将他作为一个条件去使用:

while(cin>>word)

IO库定义了一个与机器无关的iostate类型,它提供了表达流状态的完整功能:
badbit表示系统级错误,如不可恢复的读写错误;failbit表示可恢复错误,如期望读取数值缺读取了一个字符;达到文件结束的位置时,eofbitfailbit都会被置位。goodbit的值为0表示未出现错误。如果badbitfailbiteofbit任意一个被置位,检测流的条件都会失败。

输出缓冲管理

每个输出流都有一个缓冲区,用于保存程序读写的数据,如下面的代码

os<<"Hello,world";

有可能直接打印也有可能被操作系统保存到缓冲区中,随后再打印。
显式刷新缓冲区的几个方法:

cout<<"Hello,world"<<endl;  //输出字符串之后输出换行然后刷新缓冲区
cout<<"Hello,world"<<flush; //输出字符串之后直接刷新缓冲区
cout<<"Hello,world"<<ends;  //输出字符串之后输出空白字符然后刷新缓冲区

当输入流关联到输出流的时候,任何试图从输入流读取数据的操作都会先刷新关联的输出流,标准库将cout和cin关联到一起,可以使用tie()函数手动将istream和ostream关联到一起。

文件输入输出

open和close函数

在定义了一个空文件流对象之后,可以调用open将其与文件关联起来

ifstream in(ifile); //构造一个ifstream并打开给定文件
ofstream out;   //输出文件流未与任何文件相关联
out.open(ifile+".copy") //打开指定文件

文件流打开之后就会保持和对应文件的关联,如果试图将这个文件流关联到另一个文件上,必须先关闭相应的文件流。

in.close(); //关闭文件
in.open(ifile+"2"); //关联到新的文件

文件模式

每个流都有一个关联的文件模式,用来指出如何使用文件,每个文件流类型都定义了一个默认的文件模式,当我们未指定文件模式时,就使用默认模式。与ifstream关联的文件默认以in模式(读方式)打开;与ofstream相关联的文件默认以out模式(写方式)打开。文件模式如下表:

  • in 以只读方式打开
  • out 以写方式打开
  • app 每次写操作前均定位到文件末尾
  • ate 打开文件后立即定义到文件末尾
  • trunc 截断文件
  • binary 以二进制进行IO

默认情况下,当打开一个ofstream时,文件的内容会被丢弃。阻止一个ofstream清空给定文件内容的方法是同时指定app模式。

ofstream out("file1")   //文件会被清空
ofstream out("file",ofstream::app|ofstream::out)    //以app方式打开会被保存

保留ofstream打开的文件中数据的唯一办法就是指定app形式打开或者in模式打开。

string流

sstream头文件定义了三个类型来支持内存IO,这些类型可以向string写入数据,从string读取数据,就像string是个IO流一样。

当我们的工作是对整行文本进行处理,而其他的一些工作是处理行内的单个单词时,通常可以使用istringstream

string line,word;
vector<PersonInfo> people;
while(getline(cin,line)){   //不断从cin读取一行
    PersonInfo info;
    istringstream record(line);   //利用读取的这一行初始化record对象
    record>>info.name;      //从这一行中单独处理每个单词
    while(record>>word)
        info.phones.push_back(word);
    people.push_back(info);
}

在我们逐步构造输出,希望最后一行打印时,ostringstream非常有用。

for(const auto &entry:people){
    ostringstream formatted,badNums;
    for(const auto &nums:entry.phones){
        if(!valid(nums)){   //如果遇到无效电话号码,则放到badNums之后一起输出
            badNums<<" "<<nums;
        }else
        formatted<<" "<<format(nums);   //将正常的电话号码都放到formatted中
    }
    if(badNums.str().empty())
        os<<entry.name<<" "<<formatted.str()<<endl;
    else
        cerr<<"Input error: "<<entry.name<<" invalid number(s) "<<badNums.str()<<endl;
}
posted @ 2018-05-18 21:12  Frankkkk  阅读(219)  评论(0编辑  收藏  举报