C語言 文件讀寫

2023-01-11 16:26 更新

本章我們將介紹 C 程序員如何創(chuàng)建、打開、關(guān)閉文本文件或二進制文件。

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

打開文件

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

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

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

模式描述
r打開一個已有的文本文件,允許讀取文件。
w打開一個文本文件,允許寫入文件。如果文件不存在,則會創(chuàng)建一個新文件。在這里,您的程序會從文件的開頭寫入內(nèi)容。
a打開一個文本文件,以追加模式寫入文件。如果文件不存在,則會創(chuàng)建一個新文件。在這里,您的程序會在已有的文件內(nèi)容中追加內(nèi)容。
r+打開一個文本文件,允許讀寫文件。
w+打開一個文本文件,允許讀寫文件。如果文件已存在,則文件會被截斷為零長度,如果文件不存在,則會創(chuàng)建一個新文件。
a+打開一個文本文件,允許讀寫文件。如果文件不存在,則會創(chuàng)建一個新文件。讀取會從文件的開頭開始,寫入則只能是追加模式。

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

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

關(guān)閉文件

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

 int fclose( FILE *fp );

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

C 標準庫提供了各種函數(shù)來按字符或者以固定長度字符串的形式讀寫文件。

寫入文件

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

int fputc( int c, FILE *fp );

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

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

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

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

#include <stdio.h>

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

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

讀取文件

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

int fgetc( FILE * fp );

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

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

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

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

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

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

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

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


實例:

文件打開輸出就用:

 1 #include <stdio.h>
 2  
 3 int main()
 4 {
 5    FILE *fp = NULL;
 6  
 7    fp = fopen("/tmp/test.txt", "w+"); //第一個逗號前是文件位置。逗號之后是打開文件方式
 8    fprintf(fp, "This is testing for fprintf...\n");  //逗號之前是一個指針,表明往里面輸入。逗號之后fprintf是往文件里面輸入
 9    fputs("This is testing for fputs...\n", fp);
10    fclose(fp);  //記得用完關(guān)閉文件
11 }

文件打開讀?。?/p>

 1 #include <stdio.h>
 2  
 3 int main()
 4 {
 5    FILE *fp = NULL;
 6    char buff[255];
 7  
 8    fp = fopen("/tmp/test.txt", "r");
 9    fscanf(fp, "%s", buff);  //寫入的時候和平常沒有區(qū)別,還是只有字符串變量前不加‘&’,其他int、double等類型前都要加‘&’符號
10    printf("1: %s\n", buff );
11  
12    fgets(buff, 255, (FILE*)fp);  //scanf遇到空格就會斷開,gets會讀取空格,遇到換行就結(jié)束
13    printf("2: %s\n", buff );     //255是限制最大讀取內(nèi)容長度
14    
15    fgets(buff, 255, (FILE*)fp);   
16    printf("3: %s\n", buff );
17    fclose(fp);
18  
19 }

文件讀去和寫入:

文件判斷是否結(jié)尾要用feof()函數(shù)

 1 #include <stdio.h>
 2 int main()
 3 {
 4    FILE *fp = NULL;
 5    double buff;
 6    double s;
 7    int w;
 8    scanf("%lf",&s);
 9    w=s;
10    fp = fopen("coursese.txt", "w");
11    fprintf(fp,"%lf %lf %d",s,s,w);  //這個%d后面不能加'\n',因為在文件中雖然一行什么東西都沒有但是這一行確實存在,那么就不會
12    fclose(fp);                      //遇到文件結(jié)束標志。不僅換行不能交,空格也不能交
13    //即fprintf(fp,"%lf %lf %d ",s,s,w);、fprintf(fp,"%lf %lf %d ",s,s,w);  這兩種形式都錯
14    fp = fopen("coursese.txt", "r");
15    while(1){
16         if(feof(fp)) break;  
17         fscanf(fp, "%lf%lf%d", &buff,&s,&w);
18         printf("%lf %lf %d\n",buff,s,w);
19    }
20    fclose(fp);
21 }

加上%s也可以:

 1 #include <stdio.h>
 2 int main()
 3 {
 4    FILE *fp = NULL;
 5    double buff;
 6    double s;
 7    int w;
 8    char ss[55];
 9    scanf("%lf",&s);
10    scanf("%s",ss);
11    w=s;
12    fp = fopen("coursese.txt", "w");
13    fprintf(fp,"%lf %lf %d %s",s,s,w,ss);  //這個%d后面不能加'\n',因為在文件中雖然一行什么東西都沒有但是這一行確實存在,那么就不會
14    fclose(fp);                      //遇到文件結(jié)束標志。不僅換行不能交,空格也不能交
15    //即fprintf(fp,"%lf %lf %d ",s,s,w);、fprintf(fp,"%lf %lf %d ",s,s,w);  這兩種形式都錯
16    fp = fopen("coursese.txt", "r");
17    while(1){
18         if(feof(fp)) break;
19         fscanf(fp, "%lf%lf%d%s", &buff,&s,&w,ss);
20         printf("%lf %lf %d %s\n",buff,s,w,ss);
21    }
22    fclose(fp);
23 }

還有一種判斷文件結(jié)束方式:fgetc()

 但是這個函數(shù)相當于getchar(),它會在文件中吸取一個字符,這樣的話文件指針就會向后移動一位,從而導致拿出來的時候數(shù)據(jù)和進去的時候不一樣

代碼:

 1 #include <stdio.h>
 2 int main()
 3 {
 4    FILE *fp = NULL;
 5    double buff;
 6    double s;
 7    int w;
 8    char ss[55];
 9    scanf("%lf",&s);
10    scanf("%s",ss);
11    w=s;
12    fp = fopen("coursese.txt", "w");
13    fprintf(fp,"%lf %lf %d %s",s,s,w,ss);
14    fclose(fp);
15    fp = fopen("coursese.txt", "r");
16 
17    char ch;
18    while(1){
19         ch=fgetc(fp);  
20         if(ch==EOF) break;
21         fscanf(fp, "%lf%lf%d%s", &buff,&s,&w,ss);
22         printf("%lf %lf %d %s\n",buff,s,w,ss);
23    }
24    fclose(fp);
25 }

考慮到它判斷文件的方式,我們可以輸入的時候在每一條數(shù)據(jù)前面多加一個空格,來充當那個fgetc吸收的無用字符

代碼:

 1 #include <stdio.h>
 2 int main()
 3 {
 4    FILE *fp = NULL;
 5    double buff;
 6    double s;
 7    int w;
 8    char ss[55];
 9    scanf("%lf",&s);
10    scanf("%s",ss);
11    w=s;
12    fp = fopen("coursese.txt", "w");
13    fprintf(fp," %lf %lf %d %s",s,s,w,ss);  //前面多加了一個空格。也可以加其他
14    fclose(fp);
15    fp = fopen("coursese.txt", "r");
16 
17    char ch;
18    while(1){
19         ch=fgetc(fp);
20         if(ch==EOF) break;
21         fscanf(fp, "%lf%lf%d%s", &buff,&s,&w,ss);
22         printf("%lf %lf %d %s\n",buff,s,w,ss);
23    }
24    fclose(fp);
25 }


以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號