C 檔讀寫

上一章我們講解了 C 語言處理的標準輸入和輸出設備。本章我們將介紹 C 程式員如何創建、打開、關閉文本檔或二進位檔。

一個檔,無論它是文本檔還是二進位檔,都是代表了一系列的位元組。C 語言不僅提供了訪問頂層的函數,也提供了底層(OS)調用來處理存儲設備上的檔。本章將講解檔管理的重要調用。

打開檔

您可以使用 fopen( ) 函數來創建一個新的檔或者打開一個已有的檔,這個調用會初始化類型 FILE 的一個對象,類型 FILE 包含了所有用來控制流的必要的資訊。下麵是這個函數調用的原型:

FILE *fopen( const char * filename, const char * mode );

在這裏,filename 是字串,用來命名檔,訪問模式 mode 的值可以是下列值中的一個:

模式描述
r打開一個已有的文本檔,允許讀取檔。
w打開一個文本檔,允許寫入檔。如果檔不存在,則會創建一個新檔。在這裏,您的程式會從檔的開頭寫入內容。如果檔存在,則該會被截斷為零長度,重新寫入。
a打開一個文本檔,以追加模式寫入檔。如果檔不存在,則會創建一個新檔。在這裏,您的程式會在已有的檔內容中追加內容。
r+打開一個文本檔,允許讀寫檔。
w+打開一個文本檔,允許讀寫檔。如果檔已存在,則檔會被截斷為零長度,如果檔不存在,則會創建一個新檔。
a+打開一個文本檔,允許讀寫檔。如果檔不存在,則會創建一個新檔。讀取會從檔的開頭開始,寫入則只能是追加模式。

如果處理的是二進位檔,則需使用下麵的訪問模式來取代上面的訪問模式:

"rb", "wb", "ab", "rb+", "r+b", "wb+", "w+b", "ab+", "a+b"

關閉檔

為了關閉檔,請使用 fclose( ) 函數。函數的原型如下:

 int fclose( FILE *fp );

如果成功關閉檔,fclose( ) 函數返回零,如果關閉檔時發生錯誤,函數返回 EOF。這個函數實際上,會清空緩衝區中的數據,關閉檔,並釋放用於該檔的所有記憶體。EOF 是一個定義在頭檔 stdio.h 中的常量。

C 標準庫提供了各種函數來按字元或者以固定長度字串的形式讀寫檔。

寫入檔

下麵是把字元寫入到流中的最簡單的函數:

int fputc( int c, FILE *fp );

函數 fputc() 把參數 c 的字元值寫入到 fp 所指向的輸出流中。如果寫入成功,它會返回寫入的字元,如果發生錯誤,則會返回 EOF。您可以使用下麵的函數來把一個以 null 結尾的字串寫入到流中:

int fputs( const char *s, FILE *fp );

函數 fputs() 把字串 s 寫入到 fp 所指向的輸出流中。如果寫入成功,它會返回一個非負值,如果發生錯誤,則會返回 EOF。您也可以使用 int fprintf(FILE *fp,const char *format, ...) 函數來寫把一個字串寫入到檔中。嘗試下麵的實例:

注意:請確保您有可用的 tmp 目錄,如果不存在該目錄,則需要在您的電腦上先創建該目錄。

/tmp 一般是 Linux 系統上的臨時目錄,如果你在 Windows 系統上運行,則需要修改為本地環境中已存在的目錄,例如: C:\tmpD:\tmp等。

實例

#include <stdio.h> int main() { FILE *fp = NULL; fp = fopen("/tmp/test.txt", "w+"); fprintf(fp, "This is testing for fprintf...\n"); fputs("This is testing for fputs...\n", fp); fclose(fp); }

當上面的代碼被編譯和執行時,它會在 /tmp 目錄中創建一個新的檔 test.txt,並使用兩個不同的函數寫入兩行。接下來讓我們來讀取這個檔。

讀取檔

下麵是從檔讀取單個字元的最簡單的函數:

int fgetc( FILE * fp );

fgetc() 函數從 fp 所指向的輸入檔中讀取一個字元。返回值是讀取的字元,如果發生錯誤則返回 EOF。下麵的函數允許您從流中讀取一個字串:

char *fgets( char *buf, int n, FILE *fp );

函數 fgets() 從 fp 所指向的輸入流中讀取 n - 1 個字元。它會把讀取的字串複製到緩衝區 buf,並在最後追加一個 null 字元來終止字串。

如果這個函數在讀取最後一個字元之前就遇到一個換行符 '\n' 或檔的末尾 EOF,則只會返回讀取到的字元,包括換行符。您也可以使用 int fscanf(FILE *fp, const char *format, ...) 函數來從檔中讀取字串,但是在遇到第一個空格和換行符時,它會停止讀取。

實例

#include <stdio.h> int main() { FILE *fp = NULL; char buff[255]; fp = fopen("/tmp/test.txt", "r"); fscanf(fp, "%s", buff); printf("1: %s\n", buff ); fgets(buff, 255, (FILE*)fp); printf("2: %s\n", buff ); fgets(buff, 255, (FILE*)fp); printf("3: %s\n", buff ); fclose(fp); }

當上面的代碼被編譯和執行時,它會讀取上一部分創建的檔,產生下列結果:

1: This
2: is testing for fprintf...

3: This is testing for fputs...

首先,fscanf() 方法只讀取了 This,因為它在後邊遇到了一個空格。其次,調用 fgets() 讀取剩餘的部分,直到行尾。最後,調用 fgets() 完整地讀取第二行。

二進位 I/O 函數

下麵兩個函數用於二進位輸入和輸出:

size_t fread(void *ptr, size_t size_of_elements, size_t number_of_elements, FILE *a_file); size_t fwrite(const void *ptr, size_t size_of_elements, size_t number_of_elements, FILE *a_file);

這兩個函數都是用於存儲塊的讀寫 - 通常是數組或結構體。