1. 程式人生 > >C++中檔案的讀寫

C++中檔案的讀寫

在C++中如何實現檔案的讀寫?
一、ASCII 輸出 為了使用下面的方法, 你必須包含標頭檔案<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"的人, 這裡有個例子。
    int num = 150;

    char name[] = "John Doe";

    fout << "Here is a number: " << num << "/n";

    fout << "Now here is a string: " << name << "/n";

   現在儲存檔案,你必須關閉檔案,或者回寫檔案緩衝. 檔案關閉之後就不能再操作了, 所以只有在你不再操作這個檔案的時候才呼叫它,它會自動儲存檔案。回寫緩衝區會在保持檔案開啟的情況下儲存檔案, 所以只要有必要就使用它。 回寫看起來像另一次輸出, 然後呼叫方法關閉。像這樣:
    fout << flush;

    fout.close();

   現在你用文字編輯器開啟檔案,內容看起來是這樣:
  Here is a number: 150 Now here is a string: John Doe

  很簡單吧! 現在繼續檔案輸入, 需要一點技巧, 所以先確認你已經明白了流操作,對 "<<" 和">>" 比較熟悉了, 因為你接下來還要用到他們。繼續…


二、ASCII 輸入

  輸入和"cin" 流很像. 和剛剛討論的輸出流很像, 但你要考慮幾件事情。在我們開始複雜的內容之前, 先看一個文字:
    12 GameDev 15.45 L This is really awesome!

  為了開啟這個檔案,你必須建立一個in-stream物件,像這樣。
    ifstream fin("input.txt");

  現在讀入前四行. 你還記得怎麼用"<<" 操作符往流裡插入變數和符號吧?好,在 "<<" (插入)操作符之後,是">>" (提取) 操作符. 使用方法是一樣的. 看這個程式碼片段.
    int number;

    float real;

    char letter, word[8];

    fin >> number;

    fin >> word;

    fin >> real;

    fin >> letter;

  也可以把這四行讀取檔案的程式碼寫為更簡單的一行。
    fin >> number >> word >> real >> letter;

  它是如何運作的呢? 檔案的每個空白之後, ">>" 操作符會停止讀取內容, 直到遇到另一個>>操作符. 因為我們讀取的每一行都被換行符分割開(是空白字元), ">>" 操作符只把這一行的內容讀入變數。這就是這個程式碼也能正常工作的原因。但是,可別忘了檔案的最後一行。
    This is really awesome!

   如果你想把整行讀入一個char陣列, 我們沒辦法用">>"操作符,因為每個單詞之間的空格(空白字元)會中止檔案的讀取。為了驗證:
    char sentence[101];

    fin >> sentence;

  我們想包含整個句子, "This is really awesome!" 但是因為空白, 現在它只包含了"This". 很明顯, 肯定有讀取整行的方法, 它就是getline()。這就是我們要做的。
    fin.getline(sentence, 100);

  這是函式引數. 第一個引數顯然是用來接受的char陣列. 第二個引數是在遇到換行符之前,陣列允許接受的最大元素數量. 現在我們得到了想要的結果:“This is really awesome!”。 你應該已經知道如何讀取和寫入ASCII檔案了。但我們還不能罷休,因為二進位制檔案還在等著我們。


三、二進位制輸入輸出

  二進位制檔案會複雜一點, 但還是很簡單的。 首先你要注意我們不再使用插入和提取操作符(譯者注:<< 和 >> 操作符). 你可以這麼做,但它不會用二進位制方式讀寫。你必須使用read() 和write() 方法讀取和寫入二進位制檔案.

  建立一個二進位制檔案, 看下一行。
    ofstream fout("file.dat", ios::binary);

  這會以二進位制方式開啟檔案, 而不是預設的ASCII模式。

  首先從寫入檔案開始。函式write() 有兩個引數。 第一個是指向物件的char型別的指標, 第二個是物件的大小(譯者注:位元組數)。 為了說明,看例子。
    int number = 30;

    fout.write((char *)(&number), sizeof(number));

    第一個引數寫做"(char *)(&number)". 這是把一個整型變數轉為char *指標。如果你不理解,可以立刻翻閱C++的書籍,如果有必要的話。

    第二個引數寫作"sizeof(number)". sizeof() 返回物件大小的位元組數.

  就是這樣! 二進位制檔案最好的地方是可以在一行把一個結構寫入檔案。 如果說,你的結構有12個不同的成員。 用ASCII?檔案,你不得不每次一條的寫入所有成員。 但二進位制檔案替你做好了。 看這個。
    struct OBJECT

    {

      int number;

      char letter;

    } obj;

    obj.number = 15;

    obj.letter = ‘M’;

    fout.write((char *)(&obj), sizeof(obj));

  這樣就寫入了整個結構!

  接下來是輸入. 輸入也很簡單,因為read()函式的引數和 write()是完全一樣的, 使用方法也相同。
    ifstream fin("file.dat", ios::binary);

    fin.read((char *)(&obj), sizeof(obj));

  我不多解釋用法, 因為它和write()是完全相同的。二進位制檔案比ASCII檔案簡單, 但有個缺點是無法用文字編輯器編輯。

  接著, 我解釋一下ifstream 和ofstream 物件的其他一些方法作為結束.


四、更多方法

  我已經解釋了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方式開啟, 不存在則建立, 存在就覆蓋. 這裡有些額外的標誌用來改變選項。
  ios::app 新增到檔案尾 ios::ate 把檔案標誌放在末尾而非起始。 ios::trunc 預設. 截斷並覆寫檔案。 ios::nocreate 檔案不存在也不建立。 ios::noreplace 檔案存在則失敗。
檔案狀態

  我用過的唯一一個狀態函式是eof(), 它返回是否標誌已經到了檔案末尾。 我主要用在迴圈中。

  例如, 這個程式碼斷統計小寫‘e’ 在檔案中出現的次數。
    ifstream fin("file.txt");

    char ch;

    int counter;

    while (!fin.eof())

     {

       ch = fin.get();

       if (ch == ‘e’)

          counter++;

    }

     fin.close();

   我從未用過這裡沒有提到的其他方法。 還有很多方法,但是他們很少被使用。參考C++書籍或者檔案流的幫助文件來了解其他的方法。

結論

你應該已經掌握瞭如何使用ASCII檔案和二進位制檔案。有很多方法可以幫你實現輸入輸出,儘管很少有人使用他們。我知道很多人不熟悉檔案I/O操作,我希望這篇文章對你有所幫助。 每個人都應該知道. 檔案I/O還有很多顯而易見的方法,?例如包含檔案 <stdio.h>. 我更喜歡用流是因為他們更簡單。 祝所有讀了這篇文章的人好運, 也許以後我還會為你們寫些東西


無論讀寫都要包含<fstream>標頭檔案
讀:從外部檔案中將資料讀到程式中來處理,對於程式來說,是從外部讀入資料,因此定義輸入流,即定義輸入流物件:ifsteam infile,infile就是輸入流物件。 這個物件當中存放即將從檔案讀入的資料流。假設有名字為myfile.txt的檔案,存有兩行數字資料,具體方法:

    int a,b;

    ifstream infile;

    infile.open("myfile.txt");      //注意檔案的路徑

    infile>>a>>b;                   //兩行資料可以連續讀出到變數裡

    infile.close()
  如果是個很大的多行儲存的文字型檔案可以這麼讀:

    char buf[1024];                //臨時儲存讀取出來的檔案內容

    string message;

    ifstream infile;

    infile.open("myfile.js");

    if(infile.is_open())          //檔案開啟成功,說明曾經寫入過東西

    {

       while(infile.good() && !infile.eof())

       {   

        memset(buf,0,1024);   

        infile.getline(buf,1204);   

        message = buf;   

        ......                     //這裡可能對message做一些操作   

         cout<<message<<endl;

       }

       infile.close();

    }
寫:將程式中處理後的資料寫到檔案當中 對程式來說是將資料寫出去,即資料離開程式,因此定義輸出流物件ofstream outfile,outfile就是輸出流物件,這個物件用來存放將要寫到檔案當中的資料。具體做法:

    ofstream outfile;

    outfile.open("myfile.bat");  //myfile.bat是存放資料的檔名   

    if(outfile.is_open())

    {  

      outfile<<message<<endl;    //message是程式中處理的資料  

      outfile.close();

    }

    else

    {  

       cout<<"不能開啟檔案!"<<endl;

    }
c++對檔案的讀寫操作的例子
/*/從鍵盤讀入一行字元,把其中的字母依次放在磁碟檔案fa2.dat中,再把它從磁碟檔案讀入程式, 將其中的小寫字母改成大寫字母,再存入磁碟fa3.dat中*/

    #i nclude<fstream>

    #i nclude<iostream>

    #i nclude<cmath>

    using namespace std;

     //////////////從鍵盤上讀取字元的函式

     void read_save()

    {      

      char c[80];      

      ofstream outfile("f1.dat");//以輸出方工開啟檔案      

      if(!outfile)

      {                   

        cerr<<"open error!"<<endl;//注意是用的是cerr                   

        exit(1);                   

      }          

      cin.getline(c,80);//從鍵盤讀入一行字元          

      for(int i=0;c[i]!=0;i++) //對字元一個一個的處理,直到遇到'/0'為止                

         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 creat_data()

    {      

      char ch;      

      ifstream infile("f1.dat",ios::in);//以輸入的方式開啟檔案      

      if(!infile)

      {                  

         cerr<<"open error!"<<endl;                  

         exit(1);                  

       }    

      ofstream outfile("f3.dat");//定義輸出流f3.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;     

      }     

      cout<<endl;     

      infile.close();     

      outfile.close();     

     }     

    int main()

    {         

      read_save();         

      creat_data();        

      system("pause");         

      return 0;         

    }