鍍金池/ 教程/ C/ C 文件讀寫(xiě)
C 輸入 & 輸出
C 標(biāo)準(zhǔn)庫(kù) - < signal.h >
C 標(biāo)準(zhǔn)庫(kù)- < string.h >
C 循環(huán)
C 數(shù)據(jù)類型
C 字符串
C 標(biāo)準(zhǔn)庫(kù) - < stdarg.h >
C 標(biāo)準(zhǔn)庫(kù) - < math.h >
C 預(yù)處理器
C 存儲(chǔ)類
C 標(biāo)準(zhǔn)庫(kù) - < locale.h >
C 指針
C 標(biāo)準(zhǔn)庫(kù) - < assert.h >
C 結(jié)構(gòu)體
C 命令行參數(shù)
C 標(biāo)準(zhǔn)庫(kù) - < ctype.h >
C 強(qiáng)制類型轉(zhuǎn)換
C 程序結(jié)構(gòu)
C typedef
C 遞歸
C 運(yùn)算符
C 環(huán)境設(shè)置
C 函數(shù)
C 位域
C 標(biāo)準(zhǔn)庫(kù) - < stddef.h >
C 數(shù)組
C 標(biāo)準(zhǔn)庫(kù) - < errno.h >
C 標(biāo)準(zhǔn)庫(kù) -參考手冊(cè)
C 可變參數(shù)
C 變量
C 錯(cuò)誤處理
C 作用域規(guī)則
C 文件讀寫(xiě)
C 內(nèi)存管理
C 基本語(yǔ)法
C 標(biāo)準(zhǔn)庫(kù) - < stdio.h >
C 標(biāo)準(zhǔn)庫(kù) - < time.h >
C 頭文件
C 判斷
C 標(biāo)準(zhǔn)庫(kù) - < limits.h >
C 共用體
C 標(biāo)準(zhǔn)庫(kù) - < setjmp.h >
C 常量
C 標(biāo)準(zhǔn)庫(kù) - < float.h >
C 簡(jiǎn)介
C 標(biāo)準(zhǔn)庫(kù) - < stdlib.h >

C 文件讀寫(xiě)

上一章我們講解了 C 語(yǔ)言處理的標(biāo)準(zhǔn)輸入和輸出設(shè)備。本章我們將介紹 C 程序員如何創(chuàng)建、打開(kāi)、關(guān)閉文本文件或二進(jìn)制文件。

一個(gè)文件,無(wú)論它是文本文件還是二進(jìn)制文件,都是代表了一系列的字節(jié)。C 語(yǔ)言不僅提供了訪問(wèn)頂層的函數(shù),也提供了底層(OS)調(diào)用來(lái)處理存儲(chǔ)設(shè)備上的文件。本章將講解文件管理的重要調(diào)用。

打開(kāi)文件

您可以使用 fopen( ) 函數(shù)來(lái)創(chuàng)建一個(gè)新的文件或者打開(kāi)一個(gè)已有的文件,這個(gè)調(diào)用會(huì)初始化類型 FILE 的一個(gè)對(duì)象,類型 FILE 包含了所有用來(lái)控制流的必要的信息。下面是這個(gè)函數(shù)調(diào)用的原型:

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

在這里,filename 是字符串,用來(lái)命名文件,訪問(wèn)模式 mode 的值可以是下列值中的一個(gè):

模式描述
r打開(kāi)一個(gè)已有的文本文件,允許讀取文件。
w打開(kāi)一個(gè)文本文件,允許寫(xiě)入文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。在這里,您的程序會(huì)從文件的開(kāi)頭寫(xiě)入內(nèi)容。
a打開(kāi)一個(gè)文本文件,以追加模式寫(xiě)入文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。在這里,您的程序會(huì)在已有的文件內(nèi)容中追加內(nèi)容。
r+打開(kāi)一個(gè)文本文件,允許讀寫(xiě)文件。
w+打開(kāi)一個(gè)文本文件,允許讀寫(xiě)文件。如果文件已存在,則文件會(huì)被截?cái)酁榱汩L(zhǎng)度,如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。
a+打開(kāi)一個(gè)文本文件,允許讀寫(xiě)文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。讀取會(huì)從文件的開(kāi)頭開(kāi)始,寫(xiě)入則只能是追加模式。

如果處理的是二進(jìn)制文件,則需使用下面的訪問(wèn)模式來(lái)取代上面的訪問(wèn)模式:

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

關(guān)閉文件

; 為了關(guān)閉文件,請(qǐng)使用 fclose( ) 函數(shù)。函數(shù)的原型如下:

int fclose( FILE *fp );

如果成功關(guān)閉文件,fclose( ) 函數(shù)返回零,如果關(guān)閉文件時(shí)發(fā)生錯(cuò)誤,函數(shù)返回 EOF。這個(gè)函數(shù)實(shí)際上,會(huì)清空緩沖區(qū)中的數(shù)據(jù),關(guān)閉文件,并釋放用于該文件的所有內(nèi)存。EOF 是一個(gè)定義在頭文件 stdio.h 中的常量。

C 標(biāo)準(zhǔn)庫(kù)提供了各種函數(shù)來(lái)按字符或者以固定長(zhǎng)度字符串的形式讀寫(xiě)文件。

寫(xiě)入文件

下面是把字符寫(xiě)入到流中的最簡(jiǎn)單的函數(shù):

int fputc( int c, FILE *fp );

函數(shù) fputc() 把參數(shù) c 的字符值寫(xiě)入到 fp 所指向的輸出流中。如果寫(xiě)入成功,它會(huì)返回寫(xiě)入的字符,如果發(fā)生錯(cuò)誤,則會(huì)返回 EOF。您可以使用下面的函數(shù)來(lái)把一個(gè)以 null 結(jié)尾的字符串寫(xiě)入到流中:

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

函數(shù) fputs() 把字符串 s 寫(xiě)入到 fp 所指向的輸出流中。如果寫(xiě)入成功,它會(huì)返回一個(gè)非負(fù)值,如果發(fā)生錯(cuò)誤,則會(huì)返回 EOF。您也可以使用 int fprintf(FILE fp,const char format, ...) 函數(shù)來(lái)寫(xiě)把一個(gè)字符串寫(xiě)入到文件中。嘗試下面的實(shí)例:

注意:請(qǐng)確保您有可用的 /tmp 目錄,如果不存在該目錄,則需要在您的計(jì)算機(jī)上先創(chuàng)建該目錄。

#include <stdio.h>

main()
{
   FILE *fp;

   fp = fopen("/tmp/test.txt", "w+");
   fprintf(fp, "This is testing for fprintf...\n");
   fputs("This is testing for fputs...\n", fp);
   fclose(fp);
}

當(dāng)上面的代碼被編譯和執(zhí)行時(shí),它會(huì)在 /tmp 目錄中創(chuàng)建一個(gè)新的文件 test.txt,并使用兩個(gè)不同的函數(shù)寫(xiě)入兩行。接下來(lái)讓我們來(lái)讀取這個(gè)文件。

讀取文件

下面是從文件讀取單個(gè)字符的最簡(jiǎn)單的函數(shù):

int fgetc( FILE * fp );

fgetc() 函數(shù)從 fp 所指向的輸入文件中讀取一個(gè)字符。返回值是讀取的字符,如果發(fā)生錯(cuò)誤則返回 EOF。下面的函數(shù)允許您從流中讀取一個(gè)字符串:

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

函數(shù) fgets() 從 fp 所指向的輸入流中讀取 n - 1 個(gè)字符。它會(huì)把讀取的字符串復(fù)制到緩沖區(qū) buf,并在最后追加一個(gè) null 字符來(lái)終止字符串。

如果這個(gè)函數(shù)在讀取最后一個(gè)字符之前就遇到一個(gè)換行符 '\n' 或文件的末尾 EOF,則只會(huì)返回讀取到的字符,包括換行符。您也可以使用 int fscanf(FILE fp, const char format, ...) 函數(shù)來(lái)從文件中讀取字符串,但是在遇到第一個(gè)空格字符時(shí),它會(huì)停止讀取。

#include <stdio.h>

main()
{
   FILE *fp;
   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);

}

當(dāng)上面的代碼被編譯和執(zhí)行時(shí),它會(huì)讀取上一部分創(chuàng)建的文件,產(chǎn)生下列結(jié)果:

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

3: This is testing for fputs...

首先,fscanf() 方法只讀取了 This,因?yàn)樗诤筮呌龅搅艘粋€(gè)空格。其次,調(diào)用 fgets() 讀取剩余的部分,直到行尾。最后,調(diào)用 fgets() 完整地讀取第二行。

二進(jìn)制 I/O 函數(shù)

下面兩個(gè)函數(shù)用于二進(jìn)制輸入和輸出:

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);

這兩個(gè)函數(shù)都是用于存儲(chǔ)塊的讀寫(xiě) - 通常是數(shù)組或結(jié)構(gòu)體。

上一篇:C 循環(huán)下一篇:C 運(yùn)算符