IO类:

iostream 定义了用于读写流的基本类型,fstream 定义了读写命名文件的类型,sstream 定义了读写内存 string 对象的类型。

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

其中前面加了 w 的是宽字符版本的

为了支持使用宽字符的语言,标准库定义了一组类型和对象来操纵 wchar_t 类型的数据。宽字符版本的类型和函数的名字以一个 w 开始。例如:wcin,wcout,wcerr 是分别对应 cin,cout,cerr 的宽字符版对象。宽字符的类型和对象与其对应的普通 char 版本的类型定义在同一个头文件中。例如:头文件 fstream 定义了 ifstream 和 wifstream 类型。

IO类型间的关系:

标准库通过继承机制使我们能忽略不同类型的流之间的差异。利用模板,我们可以使用具有继承关系的类,而不必了解继承机制如何工作的细节。我们通常可以将一个派生类对象当作其基类对象类使用。

类型 ifstream 和 istringstream 都继承自 istream。因此,我们可以像使用 istream 对象一样来使用 ifstream 和 istringstream 对象。也就是说我们是如何使用 cin 的,就可以同样地使用这些类型的对象。例如:可以对一个 ifstream 或者 istringstream 对象调用 getline,也可以使用 >> 从一个 ifstream 或 istringstream 对象中读取数据。类似的,类型 ofstream 和 ostringstream 都继承自 ostream。因此,我们可以如何使用 cout 的,就可以同样地使用这些类型的对象。

IO对象无拷贝或赋值:

     ofstream out1, out2;
out1 = out2;//错误,不能对流对象赋值
ofstream frint(ofstream);//错误,不能初始化ofstream参数
out2 = print(out2);//错误,不能拷贝流对象

由于不能拷贝IO对象,因此我们也不能将形参或返回类型设置为流类型。进行IO操作的函数通常以引用方式传递和返回流。读写一个流会改变其状态,因此传递和返回的引用不能是 const 的。

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

注意:

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

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

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

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

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

管理条件状态:

流对象的 rdstate 成员返回一个 iostate 值,对应流的当前状态。setstate 操作将给定条件位置位,表示发生了对应错误。cleat 成员是一个重载成员:他有一个不接受参数版本,而另一个版本接受一个 iostate 类型的参数。clear 不接受参数的版本清除所有错误标志位。执行 clear() 后调用 good 会返回 true:

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

带参数的 clear 版本接受一个 iostate 值,表示流的新状态。为了复位单一的条件状态,我们首先使用 rdstate 读出当前条件状态,然后用位操作将所需复位来生成新的状态。如:cin.clear(cin.rdstate()& ~cin.failbit & ~cin.badbit);//只复位failbit和badbit,但保持eofbit不变

检查输入流的当前状态:

 #include <iostream>
using namespace std; int main(void){
int a;
cin >> a;
cout << cin.rdstate() << endl;
if(cin.rdstate() == ios::goodbit) cout << "the input is true" << endl; std::wcout.imbue(locale(locale(), "", LC_CTYPE));
if(cin.rdstate() == ios::failbit) wcout << L"输入数据类型错误,非致命错误,可清除输入缓冲区挽回!" << endl; return ;
}

通过 clear 使 cin 有效:

 #include <iostream>
using namespace std; int main(void){
int a;
cin >> a;
cout << cin.rdstate() << endl;
cin.clear(ios::goodbit);//使cin有效
cout << cin.rdstate() << endl;
return ;
}

通常当我们发现输入有错又需要改正的时候,使用 clear() 更改标记为正确后,同时也需要使用 get() 成员函数清除输入缓冲区,以达到重复输入的目的:

 #include <iostream>
using namespace std; int main(void){
int a;
while(){
cin >> a;
if(cin.fail()){
cout << "error! please input agin!" << endl;
cin.clear();
cin.get();
}else{
cout << a;
break;
}
}
return ;
}

管理输出缓冲:

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

os << "please enter a value: ";

文本串可能立即打印出来,也有可能被操作系统保存在缓冲区,随后再打印。

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

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

endl  换行并刷新缓冲区;

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

flush  仅刷新缓冲区;

unibuf 操纵符:

     cout << unitbuf;//所有输出操作后都会立即刷新缓冲区
//任何输出都立即刷新,无缓冲
cout << nounitbuf;//回到正常的缓冲方式

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

关联输入和输出流:

当一个输入流关联到一个输出流时,任何试图从输入流读取数据的操纵都会先刷新关联的输出流。标准库将 cout 和 cin 关联到一起,因此:

cin >> val;

导致 cout 的缓冲区被刷新。

tie:

tie 有两个重载版本:一个版本不带参数,返回指向输出流的指针。如果本对象关联到一个输出流,则返回的就是指向这个流的指针,如果对象未关联到流,则返回空指针。tie 的第二个版本接受一个指向 ostream 的指针,将自己关联到 ostream。即,x.tie(&o) 将流 x 关联到输出流 o。

我们既可以将一个 istream 对象关联到另一个 ostream,也可以将一个 ostream 关联到另一个 ostream:

     cin.tie(&cerr);//直接将cin由和cout关联改成和cerr关联
cin.tie(&cout);//将输入流cin关联到输出流cout
ostream *old_tie = cin.tie(nullptr);//cin不再与其他流关联
cin.tie(&cerr);//将cin关联到cerr,读取cin会刷新cerr而非cout
cin.tie(old_tie);//重建cin和cout之间的关联

注意:每个流同时最多关联到一个流,但多个流可以同时关联到一个 ostream。

IO相关1(io类/流状态)的更多相关文章

  1. 关于golang中IO相关的Buffer类浅析

    io重要的接口 在介绍buffer之前,先来认识两个重要的接口,如下边所示: type Reader interface { Read(p []byte) (n int, err error) } t ...

  2. 从零开始学C++之IO流类库(二):文件流(fstream, ifstream, ofstream)的打开关闭、流状态

    一.文件流 ofstream,由ostream派生而来,用于写文件 ifstream,由istream派生而来, 用于读文件 fstream,由iostream派生而来,用于读写文件 二.打开文件 说 ...

  3. 文件IO 相关的包:java.io文件——API

    文件IO 相关的包:java.io文件——API 1.Java.io.File类的使用(1)两种路径绝对路径:相对于当前路径:当前为 “工程名”(2)File类创建,对象为一个文件/目录,可能存在或不 ...

  4. Java之IO(十四)IO包中其它类

    转载请注明出处:http://www.cnblogs.com/lighten/p/7267553.html 1.前言 此章介绍IO包中剩余未介绍的几个流和工具类,包括LineNumberReader. ...

  5. spring源码分析-core.io包里面的类

    前些日子看<深入理解javaweb开发>时,看到第一章java的io流,发觉自己对io流真的不是很熟悉.然后看了下JDK1.7中io包的一点点代码,又看了org.springframewo ...

  6. Properties -IO相关的双列集合类

    IO相关的集合类 java.util.Properties集合 extends hashtable(淘汰) Properties类表示了一个持久的属性集.Properties可保存流中或从流中加载 P ...

  7. java.io 包下的类有哪些 + 面试题

    java.io 包下的类有哪些 + 面试题 IO 介绍 IO 是 Input/Output 的缩写,它是基于流模型实现的,比如操作文件时使用输入流和输出流来写入和读取文件等. IO 分类 传统的 IO ...

  8. Java IO相关使用

    date: 2020-06-14 14:42:22 updated: 2020-08-21 17:35:45 Java IO相关使用 1. 文件 创建 File 对象的三种方式 一个路径名:File( ...

  9. JAVA基础知识之IO——Java IO体系及常用类

    Java IO体系 个人觉得可以用"字节流操作类和字符流操作类组成了Java IO体系"来高度概括Java IO体系. 借用几张网络图片来说明(图片来自 http://blog.c ...

随机推荐

  1. eclipse安装提要

    svn 插件安装http://subclipse.tigris.org/update_1.12.x教程地址http://jingyan.baidu.com/article/f71d60376b4c57 ...

  2. 别人的Linux私房菜(2)Linux简介

    同一操作系统无法在不同硬件平台上运行.架构. Bell实验室和麻省理工学院MIT和通用电气公司GE发起了Multics计划,分时兼容系统,300以上多终端连接主机. Unics 由Multics中的人 ...

  3. MZOJ 1344 工作依赖

    这道题并不是很难,关键在于读入; 其余只需一个遍历;(考的时候傻逼兮兮的没写出来) 另外,学到了一个 isdigit()用来判断是否是0-9的数字; #include <bits/stdc++. ...

  4. idea创建spring boot+mybatis(oracle)+themeleaf项目

    1.新建项目 选择idea已经有的spring initializr next,然后填写项目命名,包名 然后next,选择所需要的依赖 然后一路next,finish,项目新建成功,然后可以删除下面的 ...

  5. RF 和 GBDT联系和区别

    1.RF 原理 用随机的方式建立一个森林,森林里面有很多的决策树,随机森林的每一棵决策树之间是没有关联的.在得到森林之后,当有一个新的输入样本进入的时候,就让森林中的每一棵决策树分别进行一下判断,看看 ...

  6. CloneZilla 恢复系统报错Waiting for device dev/disk/by-id/.....

    利用CloneZilla备份好系统,在恢复系统时候显示恢复成功,但在重启系统时出现如下错误: 出现问题的原因: 原因在于suse系统的一个新的默认设置,这个新的默认设置为存储设备由原来的名称相关改为I ...

  7. hibernate添加数据报错:Could not execute JDBC batch update

    报错如下图所示: 报错原因:在配置文件或注解里设置了字段关联,但数据却没有关联. 解决方法:我的错误是向一个多对多的关联表里插入数据,由于表中一个字段的数据是从另一张表里get到的,通过调试发现,从以 ...

  8. 2018-04-11 activity周期

    android相机开发 1.Android wifi热点连接过程 2.bindservice和AIDLhttps://blog.csdn.net/zhou_wenchong/article/detai ...

  9. (记忆化搜索)Jury Compromise (poj 1015)

    http://acm.fzu.edu.cn/problem.php?pid=1005 Description The fastfood chain McBurger owns several rest ...

  10. POJ3616--Milking Time(动态规划)

    Bessie is such a hard-working cow. In fact, she is so focused on maximizing her productivity that sh ...