为了使用下面的方法, 你必须包含头文件<fstream.h>(译者注:在标准C++中,已经使用<fstream>取代< fstream.h>,所有的C++标准头文件都是无后缀的。)。这是 <iostream.h>的一个扩展集, 提供有缓冲的文件输入输出操作. 事实上, <iostream.h> 已经被<fstream.h>包含了, 所以你不必包含所有这两个文件, 如果你想显式包含他们,那随便你。我们从文件操作类的设计开始, 我会讲解如何进行ASCII I/O操作。如果你猜是"fstream," 恭喜你答对了! 但这篇文章介绍的方法,我们分别使用"ifstream"和 "ofstream" 来作输入输出。如果你用过标准控制台流"cin"和 "cout," 那现在的事情对你来说很简单。 我们现在开始讲输出部分,首先声明一个类对象。ofstream fout; 这就可以了,不过你要打开一个文件的话, 必须像这样调用ofstream::open()。
fout.open("output.txt"); 你也可以把文件名作为构造参数来打开一个文件。
ofstream fout("output.txt"); 这是我们使用的方法, 因为这样创建和打开一个文件看起来更简单. 顺便说一句, 如果你要打开的文件不存在,它会为你创建一个, 所以不用担心文件创建的问题. 现在就输出到文件,看起来和"cout"的操作很像。对不了解控制台输出"cout"的人,这里有个例子。
<span style="font-size:12px;">int num = 150; char name[] = "John Doe"; fout << "Here is a number: " << num << "/n"; fout << "Now here is a string: " << name << "/n";</span>现在保存文件,你必须关闭文件,或者回写文件缓冲。文件关闭之后就不能再操作了, 所以只有在你不再操作这个文件的时候才调用它,它会自动保存文件。回写缓冲区会在保持文件打开的情况下保存文件, 所以只要有必要就使用它。 看起来像另一次输出, 然后调用方法关闭。像这样:
<span style="font-size:12px;">fout << flush; fout.close(); </span>现在你用文本编辑器打开文件,内容看起来是这样:
二、ASCII 输入
输入和"cin" 流很像. 和刚刚讨论的输出流很像, 但你要考虑几件事情。在我们开始复杂的内容之前, 先看一个文本:
12 GameDev 15.45 L This is really awesome! 为了打开这个文件,你必须创建一个in-stream对象,像这样。
<span style="font-size:12px;">ifstream fin("input.txt");</span>现在读入前四行. 你还记得怎么用"<<" 操作符往流里插入变量和符号吧?在 "<<" (插入)操作符之后,是">>" (提取) 操作符. 使用方法是一样的. 看这个代码片段:
<span style="font-size:12px;">int number; float real; char letter, word[8]; fin >> number;</span>
<span style="font-size:12px;">fin >> word;</span>
<span style="font-size:12px;">fin >> real; </span>
<span style="font-size:12px;">fin >> letter;</span>也可以把这四行读取文件的代码写 为更简单的一行:
<span style="font-size:12px;">fin >> number >> word >> real >> letter;</span>它是如何运作的呢? 文件的每个空白之后,">>" 操作符会停止读取内容, 直到遇到另一个>>操作符。因为我们读取的每一行都被换行符分割开(是空白字符), ">>" 操作符只把这一行的内容读入变量。这就是这个代码也能正常工作的原因。但是,可别忘了文件的最后一行。
<span style="font-size:12px;">char sentence[101]; fin >> sentence; </span>我们想包含整个句子。"This is really awesome!" 但是因为空白, 现在它只包含了"This". 很明显, 肯定有读取整行的方法, 它就是getline()。这就是我们要做的。
<span style="font-size:12px;">fin.getline(sentence, 100); </span>这是函数参数. 第一个参数显然是用来接受的char数组. 第二个参数是在遇到换行符之前,数组允许接受的最大元素数量. 现在我们得到了想要的结果:“This is really awesome!”。你应该已经知道如何读取和写入ASCII文件了。但我们还不能罢休,因为二进制文件还在等着我们。
<span style="font-size:12px;">int number = 30; fout.write((char *)(&number), sizeof(number)); </span>第一个参数写做"(char *)(&number)". 这是把一个整型变量转为char *指针。如果你不理解,可以立刻翻阅C++的书籍,如果有必要的话。第二个参数写作"sizeof(number)". sizeof() 返回对象大小的字节数. 就是这样!二进制文件最好的地方是可以在一行把一个结构写入文件。 如果说,你的结构有12个不同的成员。 用ASCII?文件,你不得不每次一条的写入所有成员。 但二进制文件替你做好了。 看这个:
<span style="font-size:12px;">struct OBJECT { int number; char letter; } obj; obj.number = 15; obj.letter = ‘M’; fout.write((char *)(&obj), sizeof(obj)); </span>这样就写入了整个结构! 接下来是输入. 输入也很简单,因为read()函数的参数和 write()是完全一样的, 使用方法也相同。
<span style="font-size:12px;">ifstream fin("file.dat", ios::binary); fin.read((char *)(&obj), sizeof(obj));</span>二进制文件比ASCII文件简单, 但有个缺点是无法用文本编辑器编辑。
四、更多方法
前面解释了ASCII文件和二进制文件, 这里是一些没有提及的底层方法。
检查文件
你已经学会了open() 和close() 方法, 不过这里还有其它你可能用到的方法。方法good() 返回一个布尔值,表示文件打开是否正确。
类似的,bad() 返回一个布尔值表示文件打开是否错误。 如果出错,就不要继续进一步的操作了。最后一个检查的方法是fail(), 和bad()有点相似, 但没那么严重。
读文件
方法get() 每次返回一个字符。方法ignore(int,char) 跳过一定数量的某个字符, 但你必须传给它两个参数。第一个是需要跳过的字符数。 第二个是一个字符, 当遇到的时候就会停止。 例子:
fin.ignore(100, ‘/n’); 会跳过100个字符,或者不足100的时候,跳过所有之前的字符,包括 ‘/n’。方法peek() 返回文件中的下一个字符, 但并不实际读取它。所以如果你用peek() 查看下一个字符, 用get() 在peek()之后读取,会得到同一个字符, 然后移动文件计数器。方法putback(char) 输入字符, 一次一个, 到流中。这个函数使用较少。
写文件
只有一个你可能会关注的方法.?那就是 put(char), 它每次向输出流中写入一个字符。
打开文件
当我们用这样的语法打开二进制文件:
ofstream fout("file.dat", ios::binary); "ios::binary"是你提供的打开选项的额外标志。默认的, 文件以ASCII方式打开, 不存在则创建, 存在就覆盖。这里有些额外的标志用来改变选项。
<span style="font-size:12px;">ios::app 添加到文件尾,每次写操作前均定位到文件末尾。 ios::ate 把文件标志放在末尾而非起始,打开文件后立即定位到文件末尾。 ios::trunc 默认. 截断并覆写文件。 ios::nocreate 文件不存在也不创建。 ios::noreplace 文件存在则失败。</span>
<span style="font-size:12px;">ifstream fin("file.txt"); char ch; int counter; while (!fin.eof()) { ch = fin.get(); if (ch == ‘e’) counter++; } fin.close();</span>还有很多方法,但是他们很少被使用。参考C++书籍或者文件流的帮助文档来了解其他的方法。
结论
你应该已经掌握了如何使用ASCII文件和二进制文件。有很多方法可以帮你实现输入输出,尽管很少有人使用他们。很多人不熟悉文件I/O操作,希望这篇文章对你有所帮助, 每个人都应该知道。文件I/O还有很多显而易见的方法。
两个C++文件读写操作的例子:
#include <iostream> #include <fstream> #include <time.h> using namespace std; int main() { srand((unsigned)time(NULL)); int K = 10; ofstream fout1("data1.txt"); ofstream fout2("data2.txt"); ofstream fout("data.txt",ofstream::out); if(fout1 == NULL || fout1 == NULL || fout == NULL) { cerr << "File written failed!" << endl; exit(1); } else { for(int i = 0; i < K; i++) { fout1 << i + rand()%10 << endl; } fout1.close(); } //ifstream fin("data1.txt"); ifstream fin; fin.open("data1.txt"); if(!fin) { cerr << "File read failed!" << endl; exit(0); } else { while(!fin.eof()) { int temp; fin >> temp; fout2 << temp << endl; cout << temp << endl; //为什么要多输出一次最后一个数??真不明白 } fin.close(); system("copy data1.txt data2.txt"); fout2.close(); } system("pause"); return 1; }
#include <iostream> #include<fstream> using namespace std; //从键盘上读取字符的函数 void readSave() { char c[80]; ofstream outFile("f1.dat"); if(!outFile) { cerr << "Open error!" << endl; exit(1); } cin.getline(c,80); //从键盘读入一行字符 for(int i = 0; c[i] != 0; i++) { if(c[i] >= 65 && c[i] <= 90 || c[i] >= 97 && c[i] <= 122)//保证输入的字符是字符 { outFile.put(c[i]); //将字母字符存入磁盘文件 cout << c[i] << " "; } cout << endl; outFile.close(); } } void createData() { char ch; ifstream inFile("f1.dat",ifstream::in); if(!inFile) { cerr << "Open error!" << endl; exit(1); } ofstream outFile("f2.dat"); //定义输出流f2.dat文件 if(!outFile) { cerr << "Open error!" << endl; exit(1); } while(inFile.get(ch)) { if(ch <= 122 && ch >= 97) { ch = ch - 32; outFile.put(ch); cout << ch; } } } int main() { readSave(); createData(); system("pasue"); return 1; }