c語言 文件讀寫
本章介紹 c 語言如何創建、打開、關閉文本文件或二進制文件。
一個文件,無論它是文本文件還是二進制文件,都是代表了一系列的字節。c 語言不僅提供了訪問頂層的函數,也提供了底層(os)調用來處理存儲設備上的文件。本章將講解文件管理的重要調用。
1. 打開文件
您可以使用 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"
2. 關閉文件
;
為了關閉文件,請使用 fclose( ) 函數。函數的原型如下:
int fclose( file *fp );
如果成功關閉文件,fclose( ) 函數返回零,如果關閉文件時發生錯誤,函數返回 eof。這個函數實際上,會清空緩沖區中的數據,關閉文件,并釋放用于該文件的所有內存。eof 是一個定義在頭文件 stdio.h 中的常量。
c 標準庫提供了各種函數來按字符或者以固定長度字符串的形式讀寫文件。
3. 寫入文件
下面是把字符寫入到流中的最簡單的函數:
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 目錄,如果不存在該目錄,則需要在您的計算機上先創建該目錄。
#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); }
當上面的代碼被編譯和執行時,它會在 /tmp 目錄中創建一個新的文件 test.txt,并使用兩個不同的函數寫入兩行。接下來讓我們來讀取這個文件。
4. 讀取文件
下面是從文件讀取單個字符的最簡單的函數:
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, ...) 函數來從文件中讀取字符串,但是在遇到第一個空格字符時,它會停止讀取。
5. 二進制 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);
這兩個函數都是用于存儲塊的讀寫 - 通常是數組或結構體。
6. 實例:
文件打開輸出就用:
#include <stdio.h> int main() { file *fp = null; fp = fopen("/tmp/test.txt", "w+"); //第一個逗號前是文件位置。逗號之后是打開文件方式 fprintf(fp, "this is testing for fprintf...\n"); //逗號之前是一個指針,表明往里面輸入。逗號之后fprintf是往文件里面輸入 fputs("this is testing for fputs...\n", fp); fclose(fp); //記得用完關閉文件 }
文件打開讀取:
#include <stdio.h> int main() { file *fp = null; char buff[255]; p = fopen("/tmp/test.txt", "r"); fscanf(fp, "%s", buff); //寫入的時候和平常沒有區別,還是只有字符串變量前不加‘&’,其他int、double等類型前都要加‘&’符號 printf("1: %s\n", buff ); fgets(buff, 255, (file*)fp); //scanf遇到空格就會斷開,gets會讀取空格,遇到換行就結束 printf("2: %s\n", buff ); //255是限制最大讀取內容長度 fgets(buff, 255, (file*)fp); printf("3: %s\n", buff ); fclose(fp); }
文件讀去和寫入:
文件判斷是否結尾要用feof()函數
#include <stdio.h> int main() { file *fp = null; double buff; double s; int w; scanf("%lf",&s); w=s; fp = fopen("coursese.txt", "w"); fprintf(fp,"%lf %lf %d",s,s,w); //這個%d后面不能加'\n',因為在文件中雖然一行什么東西都沒有但是這一行確實存在,那么就不會 fclose(fp); //遇到文件結束標志。不僅換行不能交,空格也不能交 //即fprintf(fp,"%lf %lf %d ",s,s,w);、fprintf(fp,"%lf %lf %d ",s,s,w); 這兩種形式都錯 fp = fopen("coursese.txt", "r"); while(1){ if(feof(fp)) break; fscanf(fp, "%lf%lf%d", &buff,&s,&w); printf("%lf %lf %d\n",buff,s,w); } fclose(fp); }
加上%s也可以:
#include <stdio.h> int main() { file *fp = null; double buff; double s; int w; char ss[55]; scanf("%lf",&s); scanf("%s",ss); w=s; fp = fopen("coursese.txt", "w"); fprintf(fp,"%lf %lf %d %s",s,s,w,ss); //這個%d后面不能加'\n',因為在文件中雖然一行什么東西都沒有但是這一行確實存在,那么就不會 fclose(fp); //遇到文件結束標志。不僅換行不能交,空格也不能交 //即fprintf(fp,"%lf %lf %d ",s,s,w);、fprintf(fp,"%lf %lf %d ",s,s,w); 這兩種形式都錯 fp = fopen("coursese.txt", "r"); while(1){ if(feof(fp)) break; fscanf(fp, "%lf%lf%d%s", &buff,&s,&w,ss); printf("%lf %lf %d %s\n",buff,s,w,ss); } fclose(fp); }
還有一種判斷文件結束方式:fgetc()
?但是這個函數相當于getchar(),它會在文件中吸取一個字符,這樣的話文件指針就會向后移動一位,從而導致拿出來的時候數據和進去的時候不一樣
代碼:
#include <stdio.h> int main() { file *fp = null; double buff; double s; int w; char ss[55]; scanf("%lf",&s); scanf("%s",ss); w=s; fp = fopen("coursese.txt", "w"); fprintf(fp,"%lf %lf %d %s",s,s,w,ss); fclose(fp); fp = fopen("coursese.txt", "r"); char ch; while(1){ ch=fgetc(fp); if(ch==eof) break; fscanf(fp, "%lf%lf%d%s", &buff,&s,&w,ss); printf("%lf %lf %d %s\n",buff,s,w,ss); } fclose(fp); }
考慮到它判斷文件的方式,我們可以輸入的時候在每一條數據前面多加一個空格,來充當那個fgetc吸收的無用字符
代碼:
#include <stdio.h> int main() { file *fp = null; double buff; double s; int w; char ss[55]; scanf("%lf",&s); scanf("%s",ss); w=s; fp = fopen("coursese.txt", "w"); fprintf(fp," %lf %lf %d %s",s,s,w,ss); //前面多加了一個空格。也可以加其他 fclose(fp); fp = fopen("coursese.txt", "r"); char ch; while(1){ ch=fgetc(fp); if(ch==eof) break; fscanf(fp, "%lf%lf%d%s", &buff,&s,&w,ss); printf("%lf %lf %d %s\n",buff,s,w,ss); } fclose(fp); }