下面是IO类的继承关系:

ifstream和istringstream都继承自istream。因此,我们可以在传递istream对象的地方传递ifstream和istringstream。

例如:对ifstream和istringstream对象调用getline,也可以用>>从ifstream和istringstream读取数据。

类似的ofstream和ostringstream也都继承自ostream。

IO库和头文件如下:

头文件 类型
iostream

istream,wistream从流读取数据

ostream,wostream向流写入数据

iostream,wiostream读写流

fstream

ifstream,wifstream从文件读取数据

ofstream,wofstream向文件写入数据

fstream,wfstream读写文件

sstream

istringstream,wistringstream从string读取数据

ostringstream,wostringstream向string写入数据

stringstream,wstringstream读写string

 

IO类定义了一些函数和标志,帮助我们访问和控制流的状态条件。

strm::iostate  strm是指一种IO类型,iostate是一种机器相关的类型,提供表达条件状态的完整功能。

strm::badbit  标志流已崩溃

strm::failbit  标志IO操作失败了

strm::eofbit  标志达到文件结尾

strm::goodbit 流未出现错误状态,此值保证为0

s.eof()     若流s的eofbit置位,则返回true

s.fail()     若流s的failbit或badbit置位,则返回true

s.bad()      若流s的badbit置位,则返回true

s.good()    若流s处于有效状态,则返回true

s.clear()    将流的所有条件复位,状态设置为有效,返回void

s.clear(flags)  根据flags标志位,将s中对应条件状态复位,flags类型为strm::iostate。返回void

s.setstate(flags) 根据flags标志位,将s中对应条件状态置位,flags类型为strm::iostate。返回void

s.rdstate()   返回流的当前条件状态,返回类型为strm::iostate

确定流的状态最简单的方法是把它当做条件来使用。

例如:

while(cin >> n){//while会循环检查输入的状态,成功则保持循环
...

注意:

badbit是系统级错误,如不可恢复的读写错误;一般badbit置位了流就无法继续使用。

failbit是可恢复错误,例如希望读取数值,却读取了一个字符,这种错误可以修正,流还可以继续使用。

如果达到文件结束位置,eofbit和failbit都会被置位。

googbit表示流未发生错误,值为0。

只要badbit、failbit、eofbit任意一个被置位,检查流的状态的条件会失败。

auto oldState = cin.rdstate();//记住cin的当前状态
cin.clear();//使cin有效
process_input(cin);//使用cin
cin.setstate(oldState);//将cin的状态还原

或者这样使用:

cin.clear(cin.rdstate()&~cin.failbit&~cin.badbit);//只复位failbit和badbit

进行IO操作的函数通常以引用的方式传递和返回流。

每个输出流都管理一个缓冲区,用来保存程序读写的数据。

例如:

cout << "Please input:";

字符串不一定会立刻打印出来,可能保存到缓冲区随后打印,所以有时调试时,没有输出,可以看看自己刷新了缓冲区吗?

如果程序异常终止,缓冲区是不会被刷新的,当一个程序崩溃后,它的输出很可能停留在输出缓冲区中等待打印。

会导致刷新缓冲区的原因:

  • 程序正常结束,作为main函数的return的一部分,缓冲刷新被执行。
  • 缓冲区满,需要刷新缓冲区,而后新的数据才能写入。
  • 使用操纵符:endl来显示刷新缓冲区。
  • 每个输出操作后,可以用操作符unitbuf设置流的内部状态,来清空缓冲区。默认情况下,cerr是设置unitbuf的,因此写到cerr中的数据都是立即刷新的。
  • 一个输出流被关联到另一个流,此时,当读写被关联的流时,关联到流的缓冲区会被刷新。默认cin和cerr都关联到cout,因此读cin或写cerr都会导致cout的缓冲区被刷新。

endl  换行并刷新缓冲区;

ends  插入一个空字符并刷新缓冲区;

flush  仅刷新缓冲区

使用tie函数手动关联流:

cin.tie(&cout);//将标准库cin和cout关联到一起
//oldTie指向当前关联到cin的流,如果存在的话
ostream* oldTie = cin.tie(nullptr);//cin不再与其他流关联
cin.tie(&cerr);
cin.tie(oldTie);//重建cin与cout的正常关联

 流的操作函数,以istream为例。

istream流 的操作:

1、opeartor>>操作

<<操作返回一个ostream对象的引用,所以可以连续使用


2、get( )

get( )操作:

读取单个字符

返回一个整数

 

get(char&)操作:

读取单个字符

返回一个istream对象的引用

 

3、getline( )

读取一行,遇到回车键返回istream对象的引用

getline()操作与>>的区别:  

char string1 [256],

cin.getline(string1, 256);     //get a whole line, 以'\0'结尾

cin >> string1;    //stop at the 1st blank space

4、read( )

read(buf, len)
返回一个istream对象的引用
对空白字符(包括'\n')照读不误


5、peek( ) 与 putpack()

peek:查看而不读取
putback:将一个字符添加到流

文件流总结

需要包含的头文件: <fstream> 

fstream提供了三个类,用来实现c++对文件的操作。(文件的创建,读写)。
ifstream -- 从已有的文件读

ofstream -- 向文件写内容

fstream - 打开文件供读写

支持的文件类型

实际上,文件类型可以分为两种: 文本文件和二进制文件.

文本文件保存的是可读的字符, 而二进制文件保存的只是二进制数据。利用二进制模式,你可以操作图像等文件。用文本模式,你只能读写文本文件。否则会报错。

 

string流

在sstream头文件中定义了三个类型来支持内存的IO,通过ostringstream可以向string写入数据,通过istringstream从string读取数据,通过stringstream向string读写数据

stringstream特有的操作:

sstream strm strm是一个未绑定的stringstream对象。
sstream strm(s) strm是一个stringstream对象,保存string s的拷贝。此构造函数是explicit
strm.str() 返回strm保存的字符串
strm.str(s) 将strm保存的字符串拷贝到string s中,返回void

当对整行文本并处理行内的某个单词进行处理,此时可以使用string流。

istringstream将string转换为int(通常可是直接使用to_string()和stoi(),stol(),stof(),stod()等实现string和数值类型的相互转换)

string s = "123";
istringstream isToi(s);//将s绑定到istringstream中
int a;
isToi >> a;//string输出为int
cout << a << endl;

读取某文件中人名和电话号码:

Morgan 1646464648 1164986463

drew 15465463132

struct PersonInfo{
    string name;
    vector<string> phones;//多个电话号码
}
//line保存一行的信息
string line,word;
vector<PersonInfo> people;
//读取一行
while(getline(cin,line)){
    PersonInfo info;
    istringstream iss(line);//一行的字符串绑定istringstream
    iss >> info.name;//分离出姓名,以空格分离
    while(iss >> word)info.phones.push_back(word);
    people.push_back(info);
}

输出上面的信息,但是不能输出有无效号码的人

for(const auto &info : people){
    ostringstream formatted,badNums;//保存格式化信息和无效信息
    for(const auto &num : info.phones){
        if(!vaild(num))badNums << " " << num;//如果号码无效则保存在内存badNums中
        else formatted << " " << num;//否则保存号码到formatted
    }
    if(badNums.str().empty())//没有无效号码
        cout << info.name << " " << formatted.str() << endl;
    else
        cerr << "input error:" << info.name << " " << badNums.str() << endl;
}