本 Wiki 开启了 HTTPS。但由于同 IP 的 Blog 也开启了 HTTPS,因此本站必须要支持 SNI 的浏览器才能浏览。为了兼容一部分浏览器,本站保留了 HTTP 作为兼容。如果您的浏览器支持 SNI,请尽量通过 HTTPS 访问本站,谢谢!
这里会显示出您选择的修订版和当前版本之间的差别。
两侧同时换到之前的修订记录前一修订版后一修订版 | 前一修订版 | ||
cs:programming:cpp:cpp_primer:8_io [2024/01/14 13:46] – 移除 - 外部编辑 (未知日期) 127.0.0.1 | cs:programming:cpp:cpp_primer:8_io [2024/12/09 06:42] (当前版本) – codinghare | ||
---|---|---|---|
行 1: | 行 1: | ||
+ | ======IO类====== | ||
+ | C++ Primer 笔记 第八章\\ | ||
+ | ---- | ||
+ | ====IO类==== | ||
+ | C++ 提供了全面的 //IO// 类用于读写: | ||
+ | * 支持 basic type stream 读写的 // | ||
+ | * 支持 File 读写的 // | ||
+ | * 支持 In-memory string 读写的 // | ||
+ | 以上版本都分别拥有应用于 // | ||
+ | \\ \\ | ||
+ | {{ : | ||
+ | |||
+ | ==IO类型之间的关系== | ||
+ | 概念上来讲,输入输出操作不应该因操作对象而异。C++ 通过**继承**(// | ||
+ | \\ \\ \\ | ||
+ | {{ cs: | ||
+ | \\ \\ | ||
+ | 继承也包括了操作继承。比如 // | ||
+ | |||
+ | ===IO对象不能拷贝或赋值=== | ||
+ | |||
+ | <wrap em> | ||
+ | * //stream// 类型**不能作为** parameter,同时也**不能作为**返回类型。 | ||
+ | * 由第一条,// | ||
+ | * 因为 读写操作**一定会**改变 //stream// 对象的状态,**所以传递和返回的必须是** <wrap em> | ||
+ | |||
+ | ===Condition States=== | ||
+ | IO 类提供了一系列的 flag / 函数 来判断和操作 stream 当前的状态: | ||
+ | \\ \\ {{ : | ||
+ | 一般情况下,输入的流程如下: | ||
+ | * 输入检测 | ||
+ | * 如果检测到与期望不符合输入,stream 的状态就会变为错误状态,此时之后的输入也会失败。 | ||
+ | 通常可以使用条件来判断输入是否有效: | ||
+ | <code cpp> | ||
+ | while (cin >> word) | ||
+ | //ok: read op successful | ||
+ | </ | ||
+ | ==Interrogating the State of a Stream== | ||
+ | < | ||
+ | stream 做为条件时,只能反馈 stream 是否有效。通过 flag 可以判断 stream 是出于什么原因无效。\\ \\ | ||
+ | 一共存在着 4 种类型为 strm:: | ||
+ | * //badbit//: system failure, **不可恢复**的读写操作。 | ||
+ | * // | ||
+ | * //eofbit//: reaching // | ||
+ | * // | ||
+ | 除此之外还有一些用于判定的成员函数。比如 '' | ||
+ | <code cpp> | ||
+ | while(cin) | ||
+ | </ | ||
+ | 实际上等同于 | ||
+ | <code cpp> | ||
+ | while(!cin.fail()) | ||
+ | </ | ||
+ | 所有的 flag 都支持**位运算**。 | ||
+ | </ | ||
+ | |||
+ | ==Managing the Condition State== | ||
+ | 除了以上判定函数之外,IO 类还提供了一系列的管理函数用于检测问题的类型和改变 stream 的状态: | ||
+ | * rdstate():返回当前的状态 | ||
+ | * setstate(flags): | ||
+ | * clear / clear(flags):重置当前的状态到 0 或者指定状态 | ||
+ | <code cpp> | ||
+ | // remember the current state of cin | ||
+ | auto old_state = cin.rdstate(); | ||
+ | cin.clear(); | ||
+ | process_input(cin); | ||
+ | cin.setstate(old_state); | ||
+ | </ | ||
+ | 使用位运算可以很方便的设定 flag: | ||
+ | <code cpp> | ||
+ | // rdstate returns a bit sequence which repersents the status of all bits | ||
+ | // if any bit is on (1), neg it to 0, then and it with the bit sequence | ||
+ | // so that we can turns off failbit and badbit but all other bits unchanged, in this case | ||
+ | cin.clear(cin.rdstate() & ~cin.failbit & ~cin.badbit); | ||
+ | </ | ||
+ | ===管理输出buffer=== | ||
+ | |||
+ | Buffer 的作用是存储程序的输入和输出。Buffer的由 //ostream// 管理,可以将好几种不同类型的输出组合成单一的系统级别操作。这样的方式可有效提升输出的执行效率。\\ | ||
+ | Buffer 在以下条件下会被 flush(刷新): | ||
+ | * 程序正常完成。 | ||
+ | * 缓冲区存满,需要刷新后再接受输入 | ||
+ | * 缓冲区被操作符显式清空,比如 //endl//, //flush//, //ends// 之类的操作符 | ||
+ | * //unitbuf// 操作符可以再每次输出后清空Buffer, | ||
+ | * 两个 stream 对象绑定的时候,对其中一个对象的操作会导致另一个对象的清空。例如: //cin// 和 //cout//。 | ||
+ | |||
+ | ==刷新输出buffer== | ||
+ | |||
+ | 上面提到了有三种显式刷新 buffer 的方法:// | ||
+ | * //endl// 将内容输出-> | ||
+ | * //flush// 将内容输出-> | ||
+ | * //ends// 将内容输出并在默认添加一个空字符,, | ||
+ | 除此之外还有: | ||
+ | * //unitbuf// ,用于每次输出内容后flush buffer | ||
+ | * // | ||
+ | 使用的方法: | ||
+ | <code cpp> | ||
+ | cout << endl; | ||
+ | cout << flush; | ||
+ | cout << ends; | ||
+ | cout << unitbuf; | ||
+ | cout << nounitbuf; | ||
+ | </ | ||
+ | <WRAP center round important 100%> | ||
+ | 注意:如果程序**非正常退出**(崩溃),buffer 是**不会自动 flush 的**。所以在 debug 的时候一定要注意考虑 | ||
+ | </ | ||
+ | |||
+ | |||
+ | ==绑定输入输出== | ||
+ | < | ||
+ | 之所以这么做事因为我们通常希望在进行交互的时候给予用于一个干净的 Buffer 用于输入。绑定的结果是对某一个绑定对象进行输入时,会刷新另外一个对象的 buffer。通过绑定,我们可以确保在用于输入信息之前,buffer 区的输出信息已经刷新完毕了。\\ \\ | ||
+ | < | ||
+ | 通过 '' | ||
+ | <code cpp> | ||
+ | tie(); // return a pointer that points to a tied output stream, or a nullptr if the stream is not tied. | ||
+ | tie(& | ||
+ | </ | ||
+ | 常见用法: | ||
+ | <code cpp> | ||
+ | cin.tie(& | ||
+ | ostream *old_tie = cin.tie(nullptr); | ||
+ | cin.tie(& | ||
+ | cin.tie(old_tie); | ||
+ | </ | ||
+ | 有几点需要注意的是: | ||
+ | * 绑定的过程实际上是在传递被绑定对象的指针。 | ||
+ | * 绑定关系是一对一的,但绑定对象是可以替换的。 | ||
+ | |||
+ | |||
+ | ====文件输入输出==== | ||
+ | |||
+ | 使用文件输出类的时候,我们需要调用 ''< | ||
+ | <code cpp linenums: | ||
+ | fstream fs; // creates an unbound file stream | ||
+ | fstream fs(s); // create a file stream and open the file s.s can be a string or pointer to a cstring. | ||
+ | fstream fs(s, mode); // create a fstream named s, with specified mode. | ||
+ | fs.open(s); //open the file named s. | ||
+ | fs.open(s, mode); // open the file name s, with specified mode. | ||
+ | fs.close(); // close fstream fs. | ||
+ | fs.isopen(); | ||
+ | </ | ||
+ | |||
+ | ===使用 fstream 对象=== | ||
+ | 对文件进行读写,首先需要两个对象: | ||
+ | <code cpp> | ||
+ | ifstream in(ifile); //ifstream type ' | ||
+ | ofstream out; //ofstream type ' | ||
+ | out.open(ifile + " | ||
+ | </ | ||
+ | 其次,为了确保在输出正常的情况下才进行输入,需要判定输出对象的状态: | ||
+ | <code cpp> | ||
+ | if (out) | ||
+ | </ | ||
+ | 如果输出对象不在正常状态则不会进行输入。\\ \\ | ||
+ | 需要注意的是,文件被打开以后,就会与指定的输入输出对象绑定,因此对已绑定的对象再进行 open 函数的调用会直接导致失败,并设置 //failbit// flag. 如果希望从不同的文件中读入 input,需要**先关闭当前绑定,再建立新的绑定**: | ||
+ | <code cpp> | ||
+ | in.close(); | ||
+ | in.open(" | ||
+ | </ | ||
+ | ==利用 Main 函数的参数自动输入== | ||
+ | main函数有两个 parameter: '' | ||
+ | <code cpp> | ||
+ | //Since argv[0] stores the program name, the array range for the files begins at 1 | ||
+ | for (auto p = argv + 1; p!= argv + argc; ++p) { | ||
+ | | ||
+ | //create a file with specific name and open it in each iteration | ||
+ | ifstream ifs(*p); | ||
+ | | ||
+ | //if open successful | ||
+ | if (ifs) { | ||
+ | process(ifs); | ||
+ | } | ||
+ | else | ||
+ | cerr << " couldn' | ||
+ | } | ||
+ | </ | ||
+ | 由于 '' | ||
+ | <WRAP center round tip 100%> | ||
+ | fstream 对象被销毁时会**自动**调用 close 函数。 | ||
+ | </ | ||
+ | |||
+ | ===文件模式=== | ||
+ | //open()// 函数中的第二个参数:**文件模式** ( //File Mode//) ,决定了 //fstream// 如何使用文件: | ||
+ | \\ \\ {{ : | ||
+ | 一些需要注意的规则: | ||
+ | * //out// 不能对 // | ||
+ | * 使用 //out// 的时候,// | ||
+ | * //app// (append) 模式的指定会取消 //trunc// 的效果 (意味着 //app// 和 //out// 搭配使用的话,文件是一直 //open// 的,也就是说文件中以前的内容不会被删除) | ||
+ | * //ate// 和 //binary// 适用于**任何** //fstream// 对象,也可以和其他 //mode// 联合使用。 | ||
+ | * //mode// 如果**没有显式声明**,会**自动调用默认定义**(比如 ifstream 对象是 in)。 | ||
+ | 如果在写入的时候想保留文件中的先前内容,必须指定 //app// 模式: | ||
+ | <code cpp> | ||
+ | ofstream ofs1(" | ||
+ | ofstream ofs2(" | ||
+ | </ | ||
+ | |||
+ | ====Stringstreams==== | ||
+ | Stringstream 定义了三种类型用于支持 string 在内存中的读写: | ||
+ | * // | ||
+ | * // | ||
+ | * // | ||
+ | 这些类型被定义在 ''< | ||
+ | < | ||
+ | sstream ss; //ss is an umbound stringstream. | ||
+ | sstream ss(s); //ss is an stringstream that hold a copy of the string s. | ||
+ | s.str(); // returns a copy of the string that ss holds. | ||
+ | ss.str(s); // copies the string s into ss. return void. | ||
+ | </ | ||
+ | |||
+ | ===使用 stringstream=== | ||
+ | ==istringstream 的使用实例== | ||
+ | 通常情况下,如果我们希望对某一行输入中单个的 string 进行操作,就可以使用到 // | ||
+ | <code cpp> | ||
+ | // members are public by default; see § 7.2 (p. 268) | ||
+ | struct PersonInfo { | ||
+ | string name; | ||
+ | vector< | ||
+ | }; | ||
+ | |||
+ | while (getline(cin, | ||
+ | PersonInfo info; // create an object to hold this record' | ||
+ | istringstream record(line); | ||
+ | record >> info.name; // read the name | ||
+ | while (record >> word) // read the phone numbers | ||
+ | info.phones.push_back(word); | ||
+ | people.push_back(info); | ||
+ | } | ||
+ | |||
+ | </ | ||
+ | 整个逻辑是: | ||
+ | * 使用 //getline// 获取每一段的输入,这一段的输入包括用户名与用户电话号码 | ||
+ | * 使用临时的 // | ||
+ | * 使用 // | ||
+ | * 先添加用户名到 '' | ||
+ | * 对当前 line 剩下的部分,也就是所有电话号码进行循环读取,并依次存储到成员 '' | ||
+ | * 最后将整个 '' | ||
+ | ==ostringstream 的使用实例== | ||
+ | ostringstream 可以用于希望先构造输出,最后再打印的情况: | ||
+ | <code cpp> | ||
+ | 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 | ||
+ | // '' | ||
+ | formatted << " " << format(nums); | ||
+ | } | ||
+ | |||
+ | if (badNums.str().empty()) // there were no bad numbers | ||
+ | os << entry.name << " " // print the name | ||
+ | << | ||
+ | else // otherwise, print the name and bad numbers | ||
+ | cerr << "input error: " << entry.name | ||
+ | << " invalid number(s) " << badNums.str() << endl; | ||
+ | } | ||
+ | </ | ||
+ | 上面例子希望进行三件事: | ||
+ | - 验证所有人的电话是否合法 | ||
+ | - 将合法电话与不合法电话分别进行输出 | ||
+ | - 对不合法的电话进行特别的标记 | ||
+ | 在这里,ostringstream 则是扮演了“暂存分类” 角色: | ||
+ | * 首先循环检查每个人的电话号码:外层循环是以人为单位,内层循环则是检查对应的**所有**号码 | ||
+ | * 根据号码是否有效,暂存当前号码到对应的 // | ||
+ | * 根据无效号码对应的 // |