C++相关:C++的IO库

时间:2023-03-08 21:37:18

前言

基本的IO库设施

  • istream(输入流类型),提供输入操作。
  • ostream(输出流类型),提供输出操作。
  • cin,一个istream对象,从标准输入读取数据。
  • cout,一个ostream对象,向标准输出写入数据。
  • cerr,一个ostream对象,通常用于输出程序错误消息,写入到标准错误。
  • >>运算符,用来从一个istream对象读取输入数据。
  • <<运算符,用来向一个ostream对象写入输出数据。
  • getline函数,从一个给定的istream对象读取一行数据,存入一个指定的string对象中。

IO类

三种最常用的IO头文件

  • iostream头文件定义了 控制台读写流的基本类型。
  • fstream定义了读写命名文件的类型。
  • sstream定义了读写内存string对象的类型。

IO对象不能拷贝复制

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

流的条件状态

ostream s;
bool istrue;
istrue = s.eof(); //流到达文件末尾则为true
istrue = s.fail(); //IO失败则为true
istrue = s.bad(); //流崩溃则为true
istrue = s.good(); //流处于有效状态则返回true
s.clear(); //将所有状态复位,流的状态重置为有效状态
s.clear(flags); //根据指定的flags标志位将流s的对应条件状态位复位
s.setstate(flags); //根据给定的flags标志位,将流s的对应条件状态位置位
s.rdstate(); //返回流的当前状态

管理输出缓冲

缓冲刷新(数据真正地写到输出设备和文件中)的触发机制

  • 程序正常结束,main函数return。
  • 缓冲区满,需要刷新缓冲区。
  • 使用诸如endl(换行)、flush(不附加字符)、ends(附加一个空字符)等操作符显式地刷新缓冲区。
  • 关联输入和输出流。当读写被关联的流时,关联到流的缓冲区会被刷新。如默认情况下,cerr和cin都关联到cout,因此读cin或者写cerr时都会导致cout的缓冲区被刷新。

关联输入输出流

x.tie(&o)的形式,即将流x关联到流o。

cin.tie(&cout);   //展示用,标准库默认将二者绑定
ostream *old_tie = cin.tie(nullptr); //cin不再与其他流关联
cin.tie(&cerr) //将cin与cerr关联
cin.tie(old_tie) //重建cin和cout的正常关联

文件输入输出流

三种类型:ifstream从指定文件读取数据,ofstream负责写入数据,而fstream可以读写文件。

//
string filepath;
fstream fs(filepath); //创建一个fstream文件流并打开filepath的文件。 //2.或者
fstream fs(filepath,mode); //按指定模式打开文件
/*模式
in 以读方式打开,ifstream和ftream的默认模式
out 以写的方式打开,ofstream和fstream的默认模式
app 每次写操作前均定位到文件末尾
ate 打开文件立即后立即定位到文件末尾
trunc 截断文件
binary 以二进制的形式进行IO
*/ //3.又或者
fstream fs;
fs.open(filepath); fs.close();//关闭与fs绑定的文件
fs.is_open(); //返回bool值,指出与fs关联的文件是否成功打开且尚未关闭

自动构造和析构

当一个fstream对象呗销毁时,close函数会被自动调用,如下

for(auto p = argv + ; p != argv + argc;++p)
{
ifstream input(*p); //创建输入流并打开文件
if(input) //成功打开
{
process(input); //处理文件
}
else
cerr << "couldn't open: " + string(*p);
} //每次循环input都会离开作用域而被销毁

注:1.如果ofstream对象要打开的文件不存在,那么它会自动创建一个对应字符串参数名字的文件。

2.如果文件存在,那么out模式下打开文件会丢弃已有数据,也就是说写入的内容会覆盖原有文件内容,避免这种情况的方法是指定打开的模式为app(append的缩写)。

int main()
{
string file;
while(cin >> file)
{
ofstream out(file);
// ofstream out(file,ios::app); 使用这种模式则会将写入内容添加到文件末尾
if(out)
{
string data;
cin >> data;
out << data; //会覆盖文件内容
out.close();
cout << "Success" << endl;
}
else
{
cerr << "Open: "<< file <<" failed!" << endl;
continue;
}
}
}

string流

三种类型:istringstream、ostringstream、stringstream,与文件流类似,只不过操作对象从文件变为了string。

stringstream特有的操作:

stringstream strm; //未指定绑定的对象

stringstream strm(s); //拷贝构造

string s = strm.str(); //返回strm流保存的string的拷贝

strm.str(s)   //将字符串s拷贝到strm流中

使用istringstream

1.考虑读取以下数据

格式:

人名        家庭电话   手机电话

morgan  20141441  2325255224

Jack       23232333   45525222

Lee        7944732   72255252

2.首先确定对象类

struct PersonInfo
{
string name;
vector<string> phones;
}
/*
1.C++中struct的默认访问修饰符为public。从语法上来说与class只有访问修饰符的区别
2.一般来说,struct适合构建只含有数据成员的对象,即精简的对象。
3.另外在C#中,struct属于值类型
*/

3.读取数据文件并操作

string line,word;
vector<PersonInfo> people;
//逐行从输入中读取数据,直至cin遇到文件末尾或者其他错误
while(geline(cin,line))
{
PersonInfo info;
istringstream record(line); //将记录绑定到刚读入的行
record >> info.name; //读入名字,注意string流以空格为默认分界符
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 << " " << nums;//不符合格式将其以字符串形式存入badNums
}
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; }
}

 总结

  • iostream负责控制台IO
  • fstream负责文件IO
  • stringstream负责内存中string的IO

另外,类fstream和stringstream都是继承自iostream,所以输入类都继承自istream,输出类都继承自ostream。因此,在istream对象上执行的操作同样可以应用于ifstream或者istringstream对象;ostream同理。