C++ 文件流的详解

时间:2021-12-17 10:54:38

部分内容转载:http://blog.csdn.net/kingstar158/article/details/6859379 感谢追求执着,原本想自己写,却发现了这么明白的文章。C++ 文件流的详解

C++文件流操作是基础内容之一,但是内容繁杂,其实会者不难,这里简单概述一下:

需要调用的类:

#include<fstream>

读入/输出有两种方法,第一种是流的方法。

读入:

ifstream file("InterestSettings.dat");
if (file)
{
while (!file.eof())
{
file >> alive >> threemonth >> halfyear
>> oneyear >> twoyear >> threeyear >fiveyear;
}
}
file.close();

输出:

ofstream file("InterestSettings.dat");
if (file)
{
file << alive << threemonth << halfyear
<< oneyear << twoyear << threeyear << fiveyear;
}
file.close();

上述代码中file.eof()表示文件到结尾。

一些验证流的状态的成员函数(所有都返回bool型返回值):

  • bad()如果在读写过程中出错,返回 true 。例如:当我们要对一个不是打开为写状态的文件进行写入时,或者我们要写入的设备没有剩余空间的时候。
  • fail()除了与bad() 同样的情况下会返回 true 以外,加上格式错误时也返回true ,例如当想要读入一个整数,而获得了一个字母的时候。
  • eof()如果读文件到达文件末尾,返回true。
  • good()这是最通用的:如果调用以上任何一个函数返回true 的话,此函数返回 false 。

要想重置以上成员函数所检查的状态标志,你可以使用成员函数clear(),没有参数。

ifstream 默认以ios::in的方式来进行文件的读取,也可以表述成:

fstream file("File.dat",ios::in)

关于ios::in,这是一个开关,C++ 文件操作一共有以下开关

ios::in 为输入(读)而打开文件
ios::out 为输出(写)而打开文件
ios::ate 初始位置:文件尾
ios::app 所有输出附加在文件末尾
ios::trunc 如果文件已存在则先删除该文件
ios::binary 二进制方式

这些方式是能够进行组合使用的,以“或”运算(“|”)的方式:例如

fstream DeadAccountFile("DeadAccount.dat", ios::in | ios::out | ios::binary);

除了在构造器中打开文件,也可使用.open()函数,上述代码可以改成:

fstream DeadAccountFile;
DeadAccountFile.open("DeadAccount.dat", ios::in | ios::out | ios::binary);

同理,文件使用结束时,要使用.close()函数使文件关闭

fstream setup("DeadAccount.dat", ios::out);
setup.close();

获得和设置流指针

我们可以通过使用以下成员函数来读出或配置这些指向流中读写位置的流指针:

tellg() 和 tellp()

这两个成员函数不用传入参数,返回pos_type 类型的值(根据ANSI-C++ 标准) ,就是一个整数,代表当前get 流指针的位置 (用tellg) 或 put 流指针的位置(用tellp).

seekg() 和seekp()

这对函数分别用来改变流指针get 和put的位置。两个函数都被重载为两种不同的原型:

seekg ( pos_type position );
seekp ( pos_type position );

使用这个原型,流指针被改变为指向从文件开始计算的一个绝对位置。要求传入的参数类型与函数 tellg 和tellp 的返回值类型相同。

seekg ( off_type offset, seekdir direction );
seekp ( off_type offset, seekdir direction );

使用这个原型可以指定由参数direction决定的一个具体的指针开始计算的一个位移(offset)。它可以是:

ios::beg 从流开始位置计算的位移
ios::cur 从流指针当前位置开始计算的位移
ios::end 从流末尾处开始计算的位移

流指针 get 和 put 的值对文本文件(text file)和二进制文件(binary file)的计算方法都是不同的,因为文本模式的文件中某些特殊字符可能被修改。由于这个原因,建议对以文本文件模式打开的文件总是使用seekg 和 seekp的第一种原型,而且不要对tellg 或 tellp 的返回值进行修改。对二进制文件,你可以任意使用这些函数,应该不会有任何意外的行为产生。

在二进制文件中,使用<< 和>>,以及函数(如getline)来操作符输入和输出数据,没有什么实际意义,虽然它们是符合语法的。文件流包括两个为顺序读写数据特殊设计的成员函数:write 和 read。第一个函数 (write) 是ostream 的一个成员函数,都是被ofstream所继承。而read 是istream 的一个成员函数,被ifstream 所继承。类 fstream 的对象同时拥有这两个函数。它们的原型是:

write ( char * buffer, streamsize size );
read ( char * buffer, streamsize size );

接下来我来举例说明:

输出:

fstream DeadAccountFile("DeadAccount.dat", ios::in | ios::out | ios::binary);
if (!DeadAccountFile)
{
cerr << "DataBase could not open" << endl;
exit();
}
for (int i = ; i < (int)DeadAccountArray.size(); i++)
{
DeadAccountFile.seekp(i*sizeof(DeadAccount));
DeadAccountFile.write(reinterpret_cast<const char*>(&DeadAccountArray[i]),
sizeof(DeadAccount));
}

输入:

fstream readfile("SavingAccount.dat", ios::in | ios::out | ios::binary);
if (!readfile)
{
cerr << "DataBase could not open" << endl;
exit();
}
SavingAccount SavingAccountTemp;
SavingAccountArray.clear();
for (int i = ; !readfile.eof(); i++)
{
readfile.seekg(i*sizeof(SavingAccount));
readfile.read(reinterpret_cast<char*>(&SavingAccountTemp),
sizeof(SavingAccount));
SavingAccountTemp.setinterest(alive);
SavingAccountArray.push_back(SavingAccountTemp);
}