第8章 IO库
部分IO库设施:
-
istream
:输入流类型,提供输入操作。 -
ostream
:输出流类型,提供输出操作。 -
cin
:istream
对象,从标准输入读取数据。 -
cout
:ostream
对象,向标准输出写入数据。 -
cerr
:ostream
对象,向标准错误写入数据。 -
>>
运算符:从istream
对象读取输入数据。 -
<<
运算符:向ostream
对象写入输出数据。 -
getline
函数:从istream
对象读取一行数据,写入string
对象。
IO类(The IO Classes)
头文件iostream
定义了用于读写流的基本类型,fstream
定义了读写命名文件的类型,sstream
定义了读写内存中string
对象的类型。
- 头文件
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类型和函数的名字以w
开始,如wcin
、wcout
和wcerr
分别对应cin
、cout
和cerr
。它们与其对应的普通char
版本都定义在同一个头文件中,如头文件fstream
定义了ifstream
和wifstream
类型。
可以将派生类的对象当作其基类的对象使用。
IO象无拷贝或赋值(No Copy or Assign for IO Objects)
不能拷贝或对IO对象赋值。
ofstream out1, out2;
out1 = out2; // error: cannot assign stream objects
ofstream print(ofstream); // error: can't initialize the ofstream parameter
out2 = print(out2); // error: cannot copy stream objects
由于IO对象不能拷贝,因此不能将函数形参或返回类型定义为流类型。进行IO操作的函数通常以引用方式传递和返回流。读写一个IO对象会改变其状态,因此传递和返回的引用不能是const
的。
条件状态(Condition States)
IO库条件状态:
状态 | 含义 |
---|---|
strm::iostate |
流的条件状态 |
strm::badbit |
流已崩溃 |
strm::failbit |
一个IO操作失败 |
strm::badbit |
流已崩溃 |
s.eof() |
若流s 的eofbit 置位,返回true |
s.fail() |
若流s 的failbit 或badbit 置位,返回true |
s.bad() |
若流s 的badbit 置位,返回true |
s.good() |
若流s 处于有效状态,返回true |
s.clear() |
将流s 的所有条件状态复位并将流置为有效 |
s.clear(flags) |
将流s 的条件状态置为flags |
s.rdstate() |
返回流的条件状态 |
badbit
表示系统级错误,如不可恢复的读写错误。通常情况下,一旦badbit
被置位,流就无法继续使用了。在发生可恢复错误后,failbit
会被置位,如期望读取数值却读出一个字符。如果到达文件结束位置,eofbit
和failbit
都会被置位。如果流未发生错误,则goodbit
的值为0。如果badbit
、failbit
和eofbit
任何一个被置位,检测流状态的条件都会失败。
good
函数在所有错误均未置位时返回true
。而bad
、fail
和eof
函数在对应错误位被置位时返回true
。此外,在badbit
被置位时,fail
函数也会返回true
。因此应该使用good
或fail
函数确定流的总体状态,eof
和bad
只能检测特定错误。
流对象的rdstate
成员返回一个iostate
值,表示流的当前状态。setstate
成员用于将指定条件置位(叠加原始流状态)。clear
成员的无参版本清除所有错误标志;含参版本接受一个iostate
值,用于设置流的新状态(覆盖原始流状态)。
// remember the current state of cin
auto old_state = cin.rdstate(); // remember the current state of cin
cin.clear(); // make cin valid
process_input(cin); // use cin
cin.setstate(old_state); // now reset cin to its old state
管理输出缓冲(Managing the Output Buffer)
每个输出流都管理一个缓冲区,用于保存程序读写的数据。导致缓冲刷新(即数据真正写入输出设备或文件)的原因有很多:
-
程序正常结束。
-
缓冲区已满。
-
使用操纵符(如
endl
)显式刷新缓冲区。 -
在每个输出操作之后,可以用
unitbuf
操纵符设置流的内部状态,从而清空缓冲区。默认情况下,对cerr
是设置unitbuf
的,因此写到cerr
的内容都是立即刷新的。 -
一个输出流可以被关联到另一个流。这种情况下,当读写被关联的流时,关联到的流的缓冲区会被刷新。默认情况下,
cin
和cerr
都关联到cout
,因此,读cin
或写cerr
都会刷新cout
的缓冲区。
flush
操纵符刷新缓冲区,但不输出任何额外字符。ends
向缓冲区插入一个空字符,然后刷新缓冲区。
cout << "hi!" << endl; // writes hi and a newline, then flushes the buffer
cout << "hi!" << flush; // writes hi, then flushes the buffer; adds no data
cout << "hi!" << ends; // writes hi and a null, then flushes the buffer
如果想在每次输出操作后都刷新缓冲区,可以使用unitbuf
操纵符。它令流在接下来的每次写操作后都进行一次flush
操作。而nounitbuf
操纵符则使流恢复使用正常的缓冲区刷新机制。
cout << unitbuf; // all writes will be flushed immediately
// any output is flushed immediately, no buffering
cout << nounitbuf; // returns to normal buffering
如果程序异常终止,输出缓冲区不会被刷新。
当一个输入流被关联到一个输出流时,任何试图从输入流读取数据的操作都会先刷新关联的输出流。标准库将cout
和cin
关联在一起,因此下面的语句会导致cout
的缓冲区被刷新:
交互式系统通常应该关联输入流和输出流。这意味着包括用户提示信息在内的所有输出,都会在读操作之前被打印出来。
使用tie
函数可以关联两个流。它有两个重载版本:无参版本返回指向输出流的指针。如果本对象已关联到一个输出流,则返回的就是指向这个流的指针,否则返回空指针。tie
的第二个版本接受一个指向ostream
的指针,将本对象关联到此ostream
。
cin.tie(&cout); // illustration only: the library ties cin and cout for us
// old_tie points to the stream (if any) currently tied to cin
ostream *old_tie = cin.tie(nullptr); // cin is no longer tied
// ties cin and cerr; not a good idea because cin should be tied to cout
cin.tie(&cerr); // reading cin flushes cerr, not cout
cin.tie(old_tie); // reestablish normal tie between cin and cout
每个流同时最多关联一个流,但多个流可以同时关联同一个ostream
。向tie
传递空指针可以解开流的关联。
文件输入输出(File Input and Output)
头文件fstream
定义了三个类型来支持文件IO:ifstream
从给定文件读取数据,ofstream
向指定文件写入数据,fstream
可以同时读写指定文件。
操作 | 含义 |
---|---|
fstream fstrm(s) |
打开s 文件,打开模式取决于fstream |
fstream fstrm(s, mode) |
以mode 模式打开s 文件 |
fstrm.close() |
关闭文件 |
fstrm.is_open() |
如果文件成功打开并尚未关闭,返回true |
使用文件流对象(Using File Stream Objects)
每个文件流类型都定义了open
函数,它完成一些系统操作,定位指定文件,并视情况打开为读或写模式。
创建文件流对象时,如果提供了文件名(可选),open
会被自动调用。
ifstream in(ifile); // construct an ifstream and open the given file
ofstream out; // output file stream that is not associated with any file
在C++11中,文件流对象的文件名可以是string
对象或C风格字符数组。旧版本的标准库只支持C风格字符数组。
在要求使用基类对象的地方,可以用继承类型的对象代替。因此一个接受iostream
类型引用或指针参数的函数,可以用对应的fstream
类型来调用。
可以先定义空文件流对象,再调用open
函数将其与指定文件关联。如果open
调用失败,failbit
会被置位。
对一个已经打开的文件流调用open
会失败,并导致failbit
被置位。随后试图使用文件流的操作都会失败。如果想将文件流关联到另一个文件,必须先调用close
关闭当前文件,再调用clear
重置流的条件状态(close
不会重置流的条件状态)。
当fstream
对象被销毁时,close
会自动被调用。
文件模式(File Modes)
每个流都有一个关联的文件模式,用来指出如何使用文件。
模式 | 含义 |
---|---|
in |
以读方式打开 |
out |
以写方式打开 |
app |
每次写操作前定位到文件末尾 |
ate |
打开文件后立即定位到文件末尾 |
trunc |
截断文件 |
binary |
以二进制方式读写 |
-
只能对
ofstream
或fstream
对象设定out
模式。 -
只能对
ifstream
或fstream
对象设定in
模式。 -
只有当
out
被设定时才能设定trunc
模式。 -
只要
trunc
没被设定,就能设定app
模式。在app
模式下,即使没有设定out
模式,文件也是以输出方式打开。 -
默认情况下,即使没有设定
trunc
,以out
模式打开的文件也会被截断。如果想保留以out
模式打开的文件内容,就必须同时设定app
模式,这会将数据追加写到文件末尾;或者同时设定in
模式,即同时进行读写操作。 -
ate
和binary
模式可用于任何类型的文件流对象,并可以和其他任何模式组合使用。 -
与
ifstream
对象关联的文件默认以in
模式打开,与ofstream
对象关联的文件默认以out
模式打开,与fstream
对象关联的文件默认以in
和out
模式打开。
默认情况下,打开ofstream
对象时,文件内容会被丢弃,阻止文件清空的方法是同时指定app
或in
模式。
流对象每次打开文件时都可以改变其文件模式。
ofstream out; // no file mode is set
out.open("scratchpad"); // mode implicitly out and trunc
out.close(); // close out so we can use it for a different file
out.open("precious", ofstream::app); // mode is out and app
out.close();
string
流(string
Streams)
头文件sstream
定义了三个类型来支持内存IO:istringstream
从string
读取数据,ostringstream
向string
写入数据,stringstream
可以同时读写string
的数据。
操作 | 含义 |
---|---|
sstream strm(s) |
strm 保存string s 的拷贝 |
strm.str() |
返回strm 中的string |
strm.str(s) |
将string s 拷贝至strm |
使用istringstream
(Using an istringstream
)
// members are public by default
struct PersonInfo
{
string name;
vector<string> phones;
};
string line, word; // will hold a line and word from input, respectively
vector<PersonInfo> people; // will hold all the records from the input
// read the input a line at a time until cin hits end-of-file (or another error)
while (getline(cin, line))
{
PersonInfo info; // create an object to hold this record's data
istringstream record(line); // bind record to the line we just read
record >> info.name; // read the name
while (record >> word) // read the phone numbers
info.phones.push_back(word); // and store them
people.push_back(info); // append this record to people
}
使用ostringstream
(Using ostringstream
s)
for (const auto &entry : people)
{ // for each entry in people
ostringstream formatted, badNums; // objects created on each loop
for (const auto &nums : entry.phones)
{ // for each number
if (!valid(nums))
{
badNums << " " << nums; // string in badNums
}
else
// ''writes'' to formatted's string
formatted << " " << format(nums);
}
if (badNums.str().empty()) // there were no bad numbers
os << entry.name << " " // print the name
<< formatted.str() << endl; // and reformatted numbers
else // otherwise, print the name and bad numbers
cerr << "input error: " << entry.name
<< " invalid number(s) " << badNums.str() << endl;
}