2007-04-18 17:34
文件 I/O 在C++中比烤蛋糕簡單多了。 在這篇文章里,我會詳細解釋ASCII和二進制文件的輸入輸出的每個細節(jié),值得注意的是,所有這些都是用C++完成的。
一、ASCII 輸出 為了使用下面的方法, 你必須包含頭文件<fstream.h>(譯者注:在標準C++中,已經(jīng)使用<fstream>取代< fstream.h>,所有的C++標準頭文件都是無后綴的。)。這是 <iostream.h>的一個擴展集, 提供有緩沖的文件輸入輸出操作. 事實上, <iostream.h> 已經(jīng)被<fstream.h>包含了, 所以你不必包含所有這兩個文件, 如果你想顯式包含他們,那隨便你。我們從文件操作類的設(shè)計開始, 我會講解如何進行ASCII I/O操作。如果你猜是"fstream," 恭喜你答對了! 但這篇文章介紹的方法,我們分別使用"ifstream"?和 "ofstream" 來作輸入輸出。 如果你用過標準控制臺流"cin"?和 "cout," 那現(xiàn)在的事情對你來說很簡單。 我們現(xiàn)在開始講輸出部分,首先聲明一個類對象。ofstream fout; 這就可以了,不過你要打開一個文件的話, 必須像這樣調(diào)用ofstream::open()。 fout.open("output.txt"); 你也可以把文件名作為構(gòu)造參數(shù)來打開一個文件. ofstream fout("output.txt"); 這是我們使用的方法, 因為這樣創(chuàng)建和打開一個文件看起來更簡單. 順便說一句, 如果你要打開的文件不存在,它會為你創(chuàng)建一個, 所以不用擔心文件創(chuàng)建的問題. 現(xiàn)在就輸出到文件,看起來和"cout"的操作很像。 對不了解控制臺輸出"cout"的人, 這里有個例子。 int num = 150; char name[] = "John Doe"; fout << "Here is a number: " << num << "\n"; fout << "Now here is a string: " << name << "\n"; 現(xiàn)在保存文件,你必須關(guān)閉文件,或者回寫文件緩沖. 文件關(guān)閉之后就不能再操作了, 所以只有在你不再操作這個文件的時候才調(diào)用它,它會自動保存文件。 回寫緩沖區(qū)會在保持文件打開的情況下保存文件, 所以只要有必要就使用它。回寫看起來像另一次輸出, 然后調(diào)用方法關(guān)閉。像這樣: fout << flush; fout.close(); 現(xiàn)在你用文本編輯器打開文件,內(nèi)容看起來是這樣: Here is a number: 150 Now here is a string: John Doe 很簡單吧! 現(xiàn)在繼續(xù)文件輸入, 需要一點技巧, 所以先確認你已經(jīng)明白了流操作,對 "<<" 和">>" 比較熟悉了, 因為你接下來還要用到他們。繼續(xù)… 二、ASCII 輸入 輸入和"cin" 流很像. 和剛剛討論的輸出流很像, 但你要考慮幾件事情。在我們開始復(fù)雜的內(nèi)容之前, 先看一個文本: 12 GameDev 15.45 L This is really awesome! 為了打開這個文件,你必須創(chuàng)建一個in-stream對象,?像這樣。 ifstream fin("input.txt"); 現(xiàn)在讀入前四行. 你還記得怎么用"<<" 操作符往流里插入變量和符號吧?好,?在 "<<" (插入)?操作符之后,是">>" (提取) 操作符. 使用方法是一樣的. 看這個代碼片段. int number; float real; char letter, word[8]; fin >> number; fin >> word; fin >> real; fin >> letter; 也可以把這四行讀取文件的代碼寫為更簡單的一行。 fin >> number >> word >> real >> letter; 它是如何運作的呢? 文件的每個空白之后, ">>" 操作符會停止讀取內(nèi)容, 直到遇到另一個>>操作符. 因為我們讀取的每一行都被換行符分割開(是空白字符), ">>" 操作符只把這一行的內(nèi)容讀入變量。這就是這個代碼也能正常工作的原因。但是,可別忘了文件的最后一行。 This is really awesome! 如果你想把整行讀入一個char數(shù)組, 我們沒辦法用">>"?操作符,因為每個單詞之間的空格(空白字符)會中止文件的讀取。為了驗證: char sentence[101]; fin >> sentence; 我們想包含整個句子, "This is really awesome!" 但是因為空白, 現(xiàn)在它只包含了"This". 很明顯, 肯定有讀取整行的方法, 它就是getline()。這就是我們要做的。 fin.getline(sentence, 100); 這是函數(shù)參數(shù). 第一個參數(shù)顯然是用來接受的char數(shù)組. 第二個參數(shù)是在遇到換行符之前,數(shù)組允許接受的最大元素數(shù)量. 現(xiàn)在我們得到了想要的結(jié)果:“This is really awesome!”。 你應(yīng)該已經(jīng)知道如何讀取和寫入ASCII文件了。但我們還不能罷休,因為二進制文件還在等著我們。 三、二進制 輸入輸出 二進制文件會復(fù)雜一點, 但還是很簡單的。首先你要注意我們不再使用插入和提取操作符(譯者注:<< 和 >> 操作符). 你可以這么做,但它不會用二進制方式讀寫。你必須使用read() 和write() 方法讀取和寫入二進制文件. 創(chuàng)建一個二進制文件, 看下一行。 ofstream fout("file.dat", ios::binary); 這會以二進制方式打開文件, 而不是默認的ASCII模式。首先從寫入文件開始。函數(shù)write() 有兩個參數(shù)。 第一個是指向?qū)ο蟮腸har類型的指針, 第二個是對象的大小(譯者注:字節(jié)數(shù))。 為了說明,看例子。 int number = 30; fout.write((char *)(&number), sizeof(number)); 第一個參數(shù)寫做"(char *)(&number)". 這是把一個整型變量轉(zhuǎn)為char *指針。如果你不理解,可以立刻翻閱C++的書籍,如果有必要的話。第二個參數(shù)寫作"sizeof(number)". sizeof() 返回對象大小的字節(jié)數(shù). 就是這樣! 二進制文件最好的地方是可以在一行把一個結(jié)構(gòu)寫入文件。 如果說,你的結(jié)構(gòu)有12個不同的成員。 用ASCII?文件,你不得不每次一條的寫入所有成員。 但二進制文件替你做好了。 看這個。 struct OBJECT { int number; char letter; } obj; obj.number = 15; obj.letter = ‘M’; fout.write((char *)(&obj), sizeof(obj)); 這樣就寫入了整個結(jié)構(gòu)! 接下來是輸入. 輸入也很簡單,因為read()?函數(shù)的參數(shù)和 write()是完全一樣的, 使用方法也相同。 ifstream fin("file.dat", ios::binary); fin.read((char *)(&obj), sizeof(obj)); 我不多解釋用法, 因為它和write()是完全相同的。二進制文件比ASCII文件簡單, 但有個缺點是無法用文本編輯器編輯。 接著, 我解釋一下ifstream 和ofstream 對象的其 |