文章目錄
- 開啟檔案(Open a file)
- 關閉檔案(Closing a file)
- 文字檔(Text mode files)
- 狀態標誌符的驗證(Verification of state flags)
- 獲得和設定流指標(get and put stream pointers)
- 二進位檔案(Binary files)
- 緩衝和同步(Buffers and Synchronization)
C++ 通過以下幾個類支援檔案的輸入輸出:
- ofstream: 寫操作(輸出)的檔案類 (由ostream引申而來)
- ifstream: 讀操作(輸入)的檔案類(由istream引申而來)
- fstream: 可同時讀寫操作的檔案類 (由iostream引申而來)
開啟檔案(Open a file)
對這些類的一個對象所做的第一個操作通常就是將它和一個真正的檔案聯絡起來,也就是說開啟一個檔案。被開啟的檔案在程式中由一個流對象(stream object)來表示 (這些類的一個執行個體) ,而對這個流對象所做的任何輸入輸出操作實際就是對該檔案所做的操作。
要通過一個流對象開啟一個檔案,我們使用它的成員函數open():
void open (const char * filename, openmode mode);這裡filename 是一個字串,代表要開啟的檔案名稱,mode 是以下標誌符的一個組合:
| ios::in |
為輸入(讀)而開啟檔案 |
| ios::out |
為輸出(寫)而開啟檔案 |
| ios::ate |
初始位置:檔案尾 |
| ios::app |
所有輸出附加在檔案末尾 |
| ios::trunc |
如果檔案已存在則先刪除該檔案 |
| ios::binary |
二進位方式 |
這些標識符可以被組合使用,中間以”或”操作符(|)間隔。例如,如果我們想要以二進位方式開啟檔案”example.bin” 來寫入一些資料,我們可以通過以下方式調用成員函數open()來實現:
ofstream file;
file.open ("example.bin", ios::out | ios::app | ios::binary); ofstream, ifstream 和 fstream所有這些類的成員函數open 都包含了一個預設開啟檔案的方式,這三個類的預設各不相同:
| 類 |
參數的預設 |
| ofstream |
ios::out | ios::trunc |
| ifstream |
ios::in |
| fstream |
ios::in | ios::out |
只有當函數被調用時沒有聲明方式參數的情況下,預設值才會被採用。如果函數被調用時聲明了任何參數,預設值將被完全改寫,而不會與調用參數組合。
由於對類ofstream, ifstream 和 fstream 的對象所進行的第一個操作通常都是開啟檔案,這些類都有一個建構函式可以直接調用open 函數,並擁有同樣的參數。這樣,我們就可以通過以下方式進行與上面同樣的定義對象和開啟檔案的操作:
ofstream file ("example.bin", ios::out | ios::app | ios::binary);兩種開啟檔案的方式都是正確的。
你可以通過調用成員函數is_open()來檢查一個檔案是否已經被順利的開啟了:
bool is_open();它返回一個布爾(bool)值,為真(true)代表檔案已經被順利開啟,假( false )則相反。
關閉檔案(Closing a file)
當檔案讀寫操作完成之後,我們必須將檔案關閉以使檔案重新變為可訪問的。關閉檔案需要調用成員函數close(),它負責將緩衝中的資料排放出來並關閉檔案。它的格式很簡單:
void close ();這個函數一旦被調用,原先的流對象(stream object)就可以被用來開啟其它的檔案了,這個檔案也就可以重新被其它的進程(process)所有訪問了。
為防止流對象被銷毀時還聯絡著開啟的檔案,解構函式(destructor)將會自動調用關閉函數close。
文字檔(Text mode files)
類ofstream, ifstream 和fstream 是分別從ostream, istream 和iostream 中引申而來的。這就是為什麼 fstream 的對象可以使用其父類的成員來訪問資料。
一般來說,我們將使用這些類與同控制台(console)互動同樣的成員函數(cin 和 cout)來進行輸入輸出。如下面的例題所示,我們使用重載的插入操作符<<:
// writing on a text file #include <fiostream.h>int main () { ofstream examplefile (”example.txt”); if (examplefile.is_open()) { examplefile << “This is a line./n”; examplefile << “This is another line./n”; examplefile.close(); } return 0; } |
file example.txt This is a line. This is another line. |
從檔案中讀入資料也可以用與 cin的使用同樣的方法:
// reading a text file #include <iostream.h> #include <fstream.h> #include <stdlib.h>int main () { char buffer[256]; ifstream examplefile (”example.txt”); if (! examplefile.is_open()) { cout << “Error opening file”; exit (1); } while (! examplefile.eof() ) { examplefile.getline (buffer,100); cout << buffer << endl; } return 0; } |
This is a line. This is another line. |
上面的例子讀入一個文字檔的內容,然後將它列印到螢幕上。注意我們使用了一個新的成員函數叫做eof ,它是ifstream 從類 ios 中繼承過來的,當到達檔案末尾時返回true 。
狀態標誌符的驗證(Verification of state flags)
除了eof()以外,還有一些驗證流的狀態的成員函數(所有都返回bool型傳回值):
- bad()如果在讀寫過程中出錯,返回 true 。例如:當我們要對一個不是開啟為寫狀態的檔案進行寫入時,或者我們要寫入的裝置沒有剩餘空間的時候。
- fail()除了與bad() 同樣的情況下會返回 true 以外,加上格式錯誤時也返回true ,例如當想要讀入一個整數,而獲得了一個字母的時候。
- eof()如果讀檔案到達檔案末尾,返回true。
- good()這是最通用的:如果調用以上任何一個函數返回true 的話,此函數返回 false 。
要想重設以上成員函數所檢查的狀態標誌,你可以使用成員函數clear(),沒有參數。
獲得和設定流指標(get and put stream pointers)
所有輸入/輸出流對象(i/o streams objects)都有至少一個流指標:
- ifstream, 類似istream, 有一個被稱為get pointer的指標,指向下一個將被讀取的元素。
- ofstream, 類似 ostream, 有一個指標 put pointer ,指向寫入下一個元素的位置。
- fstream, 類似 iostream, 同時繼承了get 和 put
我們可以通過使用以下成員函數來讀出或配置這些指向流中讀寫位置的流指標:
- 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 的傳回值進行修改。對二進位檔案,你可以任意使用這些函數,應該不會有任何意外的行為產生。
以下例子使用這些函數來獲得一個二進位檔案的大小:
// obtaining file size #include <iostream.h> #include <fstream.h>const char * filename = “example.txt”;int main () { long l,m; ifstream file (filename, ios::in|ios::binary); l = file.tellg(); file.seekg (0, ios::end); m = file.tellg(); file.close(); cout << “size of ” << filename; cout << ” is ” << (m-l) << ” bytes./n”; return 0; } |
size of example.txt is 40 bytes. |
二進位檔案(Binary files)
在二進位檔案中,使用<< 和>>,以及函數(如getline)來操作符輸入和輸出資料,沒有什麼實際意義,雖然它們是符合文法的。
文 件流包括兩個為順序讀寫資料特殊設計的成員函數:write 和 read。第一個函數 (write) 是ostream 的一個成員函數,都是被ofstream所繼承。而read 是istream 的一個成員函數,被ifstream 所繼承。類 fstream 的對象同時擁有這兩個函數。它們的原型是:
write ( char * buffer, streamsize size );
read ( char * buffer, streamsize size );
這裡 buffer 是一塊記憶體的地址,用來儲存或讀出資料。參數size 是一個整數值,表示要從緩衝(buffer)中讀出或寫入的字元數。
// reading binary file #include <iostream> #include <fstream.h>const char * filename = “example.txt”;int main () { char * buffer; long size; ifstream file (filename, ios::in|ios::binary|ios::ate); size = file.tellg(); file.seekg (0, ios::beg); buffer = new char [size]; file.read (buffer, size); file.close();cout << “the complete file is in a buffer”;delete[] buffer; return 0; } |
The complete file is in a buffer |
緩衝和同步(Buffers and Synchronization)
當 我們對檔案流進行操作的時候,它們與一個streambuf 類型的緩衝(buffer)聯絡在一起。這個緩衝(buffer)實際是一塊記憶體空間,作為流(stream)和物理檔案的媒介。例如,對於一個輸出資料流, 每次成員函數put (寫一個單個字元)被調用,這個字元不是直接被寫入該輸出資料流所對應的物理檔案中的,而是首先被插入到該流的緩衝(buffer)中。
當緩衝被排放出來(flush)時,它裡面的所有資料或者被寫入物理媒質中(如果是一個輸出資料流的話),或者簡單的被抹掉(如果是一個輸入資料流的話)。這個過程稱為同步(synchronization),它會在以下任一情況下發生:
- 當檔案被關閉時: 在檔案被關閉之前,所有還沒有被完全寫出或讀取的緩衝都將被同步。
- 當緩衝buffer 滿時:緩衝Buffers 有一定的空間限制。當緩衝滿時,它會被自動同步。
- 控制符明確指明:當遇到流中某些特定的控制符時,同步會發生。這些控制符包括:flush 和endl。
- 明確調用函數sync(): 調用成員函數sync() (無參數)可以引發立即同步。這個函數返回一個int 值,等於-1 表示流沒有聯絡的緩衝或操作失敗。