文件操作(2)

  • 1.二进制模式读取文本文件
  • 2.使用二进制读写其他类型内容
  • 3.fstream类
  • 4.文件的随机存取
    • 文件指针的获取
    • 文件指针的移动

1.二进制模式读取文本文件

用二进制方式打开文本存储的文件时,也可以读取其中的内容,因为文本文件本质上是存储字符类型数据。这种方式读取文件内容我们需要用到string类型的方法:

int main(){ifstream mytest("test.txt",ios::in|ios::binary); // ios::in是默认参数,可以不写if(mytest.is_open()){cout<<"打开文件失败"<<endl;return 0;}// 二进制文件读取后需要用正确的接收格式接收// string out((istreambuf_iterator(mytest)), (istreambuf_iterator())) // 直接使用string的构造函数对out进行赋值string out;out.assign((istreambuf_iterator<char>(mytest)), (istreambuf_iterator<char>())); // 使用assign函数赋值cout<<out;// 关闭文件mytest.close();}

istreambuf_iterator会迭代访问文件内容,读取完成后out会存贮整个文件的内容,不需要一行一行读取:

2.使用二进制读写其他类型内容

有时候我们也需要借由二进制模式处理一些非文本形式的简单数据,比如数字数组。这样的类型处理方法和上一节用结构体类型读写二进制文件很相似:

int main(){ofstream mytest("nums.txt",ios::app|ios::binary); // 读取二进制文件if(mytest.is_open()){// 二进制文件读取后需要用正确的接收格式接收int numbers[]={10,15,20,30,55,67};mytest.write(reinterpret_cast<const char*>(numbers),sizeof(numbers));}else{cout<<"打开文件失败"<<endl;return 0;}// 关闭文件mytest.close();}

这里我们将一个整形数组以二进制形式写成了二进制文件,这里的reinterpret_cast(numbers)是C++指针类型的强制转换,等同于(const char*)numbers。
当我们想要读取时,也用类似的办法处理。值得注意的是,如果我们知道数组的具体大小,读取内容的任务又会简单许多:

int main(){ifstream mytest("nums.txt",ios::in|ios::binary);if(mytest.is_open()){int numbers[6];mytest.read(reinterpret_cast<char*>(numbers), sizeof(numbers));// 查看内容是否正确存储到整形数组里for(int i=0;i<=5;i++){cout<<numbers[i]<<" ";}}else{cout<<"打开文件失败"<<endl;return 0;}// 关闭文件mytest.close();}// 输出为:10 15 20 30 55 67 

如果换成长度已知的结构体数组,也可以这样处理。
如果我们不知道数组的具体长度,也可以一个一个读出文件的存储内容:

 int main(){ifstream mytest("nums.txt",ios::in|ios::binary); // 读取二进制文件if(mytest.is_open()){int number;while(mytest.read((char*)&number,sizeof(int))){cout<<number<<" ";}}else{cout<<"打开文件失败"<<endl;return 0;}// 关闭文件mytest.close();}

输出结果同上例一样。

3.fstream类

在C语言中,文件操作只有文件指针,没有输入输出流的区别,C++是在C基础上将输入和输出分别封装成类,ifstream类用于读文件,ofstream用于写入文件。但是C++也保留了同时可以完成读写的类:fstream。它的使用方法与ifstream和ofstream完全相似,如果我们想写如文件:

int main(){fstream mytest("test.txt"); if(mytest.is_open()){mytest<<"这是另一个测试\n";}else{cout<<"打开文件失败"<<endl;return 0;}// 关闭文件mytest.close();}

如果我们想读取文件:

int main(){fstream mytest("test.txt"); if(mytest.is_open()){string out;while(mytest>>out){cout<<out<<endl;}}else{cout<<"打开文件失败"<<endl;return 0;}// 关闭文件mytest.close();}// 输出为:这是一个测试//我们尝试连续输入内容//继续测试这是一个测试//我们尝试连续输入内容//这是一个测试//我们尝试连续输入内容//这是另一个测试

我们以打开文本文件为例,展示了fstream类用法,操作二进制文件的方法也可以直接照搬ifstream和ofstream类。但是fstream仍有一些细节需要我们注意,fstream类的默认写参数是ios::out和ios::in,至于具体执行那种操作会根据后面的代码进行确定。写入文件时默认参数ios::out参数在没有文件时会创建文件,但再有文件时是默认在文件最后写内容,类似于ios::app。fstream打开文件时还有一些参数可供使用:

ios::ate 以定位到文件末尾的方式打开文件
ios::in|ios::out 以读写方式打开文件
ios::out|ios::truct 如果文件存在,则截断文件重新写入内容,类似于ofstream类的out模式。

通常情况下,规范的编程通常在需要写文件的时候使用ofstream,需要读的时候用ifstream,即需要读又需要写的时候再使用fstream类。在Linux平台下,读和写有严格的权限控制,为了方便管理,我们调用的权限应当尽量少。举个例子,如果我们只需要读取文件,那么即使我们有读写的权限,也应当以只读方式打开文件。

4.文件的随机存取

之前我们介绍了文件的写入都是在文件的末尾或删除文件内容后再写入内容,而读文件都是从文件的开头进行的。这是因为读写文件都是从文件位置指针1处开始的,我们之前的操作文件的方式,文件指针都会在文件的最开始或最末尾。实现文件的随机存取,关键就在于调整文件指针所在位置。

文件指针的获取

获取文件指针位置的方法,输入流ofstream类为成员函数是 tellp();输出流ifstream类为成员函数是 tellg();fstream类两个成员函数都有且效果完全相同。以写文件为例:

int main(){ofstream mytest("test1.txt",ios::app); if(mytest.is_open()){cout<<mytest.tellp()<<endl;mytest<<"以写入文本为例\n";cout<<mytest.tellp()<<endl;}else{cout<<"打开文件失败"<<endl;return 0;}// 关闭文件mytest.close();}// 输出为:0//22

读取文件同时获取文件指针的方法与写入相同:

// Student结构体上节内容中有所定义int main(){ifstream mytest("test.doc",ios::app|ios::binary); if(mytest.is_open()){cout<<mytest.tellg()<<endl;Student child;while(mytest.read((char*)&child,sizeof(child))){cout<<child.name<<" "<<child.age<<" "<<child.sex<<endl;cout<<"mytest.tellg()="<<mytest.tellg()<<endl;}}else{cout<<"打开文件失败"<<endl;return 0;}// 关闭文件mytest.close();}// 输出为:0// ZhangSan 15 m//mytest.tellg()=40//LiSi 20 m//mytest.tellg()=80

可以看到,在二进制文件中获取文件指针位置的方法与文本文件相同。此外,给大家说个冷知识,使用ios::app打开文件除了可以在文件末尾追加内容,也可以用于读取文件信息,文件指针位置会根据任务自动进行调整。不过为了规范编程尽量不要这样乱用。
fstream类在这里就不做展示了,用法完全相同大家可以自行尝试。

PS:文件只有一个位置指针,并非同时拥有读指针和写指针。

文件指针的移动

文件的读和写都是在当前文件指针的位置往后进行的,也就是说如果我们能够移动文件的位置指针就可以做到调整读写数据的位置。C++中为我们提供了这样的方法,ifstream类使用seekg()成员函数移动文件指针,ofstream类使用seekp()成员函数移动文件指针,fstream类依旧是两者都可用,效果相同。
seekp和seekg有两个常用的重载,第一种为:

seekg(ios::beg) 将文件指针移动到0位置
seekg(ios::end) 将文件指针移动到末尾
seekg(128) 将文件指针移动到指定位置,这里指定文件指针移动到128位置
注:将seekg换成seekp效果完全类似

我们看个例子:

int main(){ifstream mytest("test.txt",ios::in); if(mytest.is_open()){mytest.seekg(29);cout<<mytest.tellg()<<endl;string out;while(mytest>>out){cout<<out<<" "<<"当前位置为:"<<mytest.tellg()<<endl;}}else{cout<<"打开文件失败"<<endl;return 0;}mytest.close();}

输出结果为:

另外需要注意,由于一个中文占3个字节,因此我需要注意文件指针所在位置不能在某个中文字的内部。假如使用mytest.seekg(28),输出内容就会出现异常。
另一种常用seek方法的重载有两个参数:

seekg(10,ios::beg) 文件指针从0位置开始向后移动10字节
seekg(-3,ios::end) 文件指针从末尾开始向前移动3字节
seekg(5,ios::cur) 文件指针从当前位置开始向后移动5字节
注:将seekg换成seekp效果完全类似

还是看个例子:

// student类与test.doc文化部在上节中已经定义和创建int main(){fstream mytest("test.doc",ios::in | ios::out | ios::binary ); // 使用读写模式打开二进制文件test.docif(mytest.is_open()){cout<<mytest.tellg()<<endl;Student child;while(mytest.read((char*)&child,sizeof(child))){cout<<child.name<<" "<<child.age<<" "<<child.sex<<endl;cout<<"mytest.tellg()="<<mytest.tellg()<<endl;}// 清除文件状态流的标志mytest.clear();cout<<"\n以上内容作为对比\n"<<endl;mytest.seekg(-40,ios::cur);child=Student{"WangWu",22,'w'};mytest.write((const char*)&child,sizeof(Student));mytest.seekg(ios::beg);while(mytest.read((char*)&child,sizeof(child))){cout<<child.name<<" "<<child.age<<" "<<child.sex<<endl;cout<<"mytest.tellg()="<<mytest.tellg()<<endl;}mytest.close();}else{cout<<"打开文件失败"<<endl;return 0;}}

在这段代码中,大家可能会不理解mytest.clear()的用意,在我们使用while循环读取文件的全部内容后,failbite会被设置为true2,这标志会让计算机认为文件遇到错误,后面就无法正常对文件进行操作了。使用clear函数可以清除这些标志,后面才能继续操作文件。
输出是这样的:

可以看到我们虽然移动了文件指针,在再定位置添加了内容,但是原有的内容却是被覆盖掉了。如果我们想要保留原来的内容,需要将插入位置后面的内容向后移动若干单位,实现方法类似于在数组中插入一个数。

本节我们继续学习了C++操作文件的方法,其中,改变文件指针的位置是我们学习的重点。在实际应用中,我们经常会遇到需要从指定位置读取和写入内容的任务,希望大家能够掌握本节内容。


  1. 在C++中,文件位置指针的指向就是文件中进行读取或写入操作时的初始位置。 ↩︎

  2. 除了我们用过的方法(如is_open())外,文件流还有很多比较实用的方法,如使用eof()方法判断是否到达文件末尾,使用good函数判断文件写入是否成功,使用get函数按字符提取文本内容等,在下一节还会具体介绍。 ↩︎