c語言 文件讀寫
本章介紹 c 語言如何創(chuàng)建、打開、關(guān)閉文本文件或二進(jìn)制文件。
一個(gè)文件,無論它是文本文件還是二進(jìn)制文件,都是代表了一系列的字節(jié)。c 語言不僅提供了訪問頂層的函數(shù),也提供了底層(os)調(diào)用來處理存儲設(shè)備上的文件。本章將講解文件管理的重要調(diào)用。
1. 打開文件
您可以使用 fopen( ) 函數(shù)來創(chuàng)建一個(gè)新的文件或者打開一個(gè)已有的文件,這個(gè)調(diào)用會(huì)初始化類型 file 的一個(gè)對象,類型 file 包含了所有用來控制流的必要的信息。下面是這個(gè)函數(shù)調(diào)用的原型:
file *fopen( const char * filename, const char * mode );
在這里,filename 是字符串,用來命名文件,訪問模式 mode 的值可以是下列值中的一個(gè):
模式 | 描述 |
---|---|
r | 打開一個(gè)已有的文本文件,允許讀取文件。 |
w | 打開一個(gè)文本文件,允許寫入文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。在這里,您的程序會(huì)從文件的開頭寫入內(nèi)容。 |
a | 打開一個(gè)文本文件,以追加模式寫入文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。在這里,您的程序會(huì)在已有的文件內(nèi)容中追加內(nèi)容。 |
r+ | 打開一個(gè)文本文件,允許讀寫文件。 |
w+ | 打開一個(gè)文本文件,允許讀寫文件。如果文件已存在,則文件會(huì)被截?cái)酁榱汩L度,如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。 |
a+ | 打開一個(gè)文本文件,允許讀寫文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。讀取會(huì)從文件的開頭開始,寫入則只能是追加模式。 |
如果處理的是二進(jìn)制文件,則需使用下面的訪問模式來取代上面的訪問模式:
"rb", "wb", "ab", "rb+", "r+b", "wb+", "w+b", "ab+", "a+b"
2. 關(guān)閉文件
;
為了關(guān)閉文件,請使用 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)庫提供了各種函數(shù)來按字符或者以固定長度字符串的形式讀寫文件。
3. 寫入文件
下面是把字符寫入到流中的最簡單的函數(shù):
int fputc( int c, file *fp );
函數(shù) fputc() 把參數(shù) c 的字符值寫入到 fp 所指向的輸出流中。如果寫入成功,它會(huì)返回寫入的字符,如果發(fā)生錯(cuò)誤,則會(huì)返回 eof。您可以使用下面的函數(shù)來把一個(gè)以 null 結(jié)尾的字符串寫入到流中:
int fputs( const char *s, file *fp );
函數(shù) fputs() 把字符串 s 寫入到 fp 所指向的輸出流中。如果寫入成功,它會(huì)返回一個(gè)非負(fù)值,如果發(fā)生錯(cuò)誤,則會(huì)返回 eof。您也可以使用 int fprintf(file *fp,const char *format, ...) 函數(shù)來寫把一個(gè)字符串寫入到文件中。嘗試下面的實(shí)例:
注意:請確保您有可用的 /tmp 目錄,如果不存在該目錄,則需要在您的計(jì)算機(jī)上先創(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); }
當(dāng)上面的代碼被編譯和執(zhí)行時(shí),它會(huì)在 /tmp 目錄中創(chuàng)建一個(gè)新的文件 test.txt,并使用兩個(gè)不同的函數(shù)寫入兩行。接下來讓我們來讀取這個(gè)文件。
4. 讀取文件
下面是從文件讀取單個(gè)字符的最簡單的函數(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 字符來終止字符串。
如果這個(gè)函數(shù)在讀取最后一個(gè)字符之前就遇到一個(gè)換行符 '\n' 或文件的末尾 eof,則只會(huì)返回讀取到的字符,包括換行符。您也可以使用 int fscanf(file *fp, const char *format, ...) 函數(shù)來從文件中讀取字符串,但是在遇到第一個(gè)空格字符時(shí),它會(huì)停止讀取。
5. 二進(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ù)都是用于存儲塊的讀寫 - 通常是數(shù)組或結(jié)構(gòu)體。
6. 實(shí)例:
文件打開輸出就用:
#include <stdio.h> int main() { file *fp = null; fp = fopen("/tmp/test.txt", "w+"); //第一個(gè)逗號前是文件位置。逗號之后是打開文件方式 fprintf(fp, "this is testing for fprintf...\n"); //逗號之前是一個(gè)指針,表明往里面輸入。逗號之后fprintf是往文件里面輸入 fputs("this is testing for fputs...\n", fp); fclose(fp); //記得用完關(guān)閉文件 }
文件打開讀?。?/p>
#include <stdio.h> int main() { file *fp = null; char buff[255]; p = fopen("/tmp/test.txt", "r"); fscanf(fp, "%s", buff); //寫入的時(shí)候和平常沒有區(qū)別,還是只有字符串變量前不加‘&’,其他int、double等類型前都要加‘&’符號 printf("1: %s\n", buff ); fgets(buff, 255, (file*)fp); //scanf遇到空格就會(huì)斷開,gets會(huì)讀取空格,遇到換行就結(jié)束 printf("2: %s\n", buff ); //255是限制最大讀取內(nèi)容長度 fgets(buff, 255, (file*)fp); printf("3: %s\n", buff ); fclose(fp); }
文件讀去和寫入:
文件判斷是否結(jié)尾要用feof()函數(shù)
#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); //這個(gè)%d后面不能加'\n',因?yàn)樵谖募须m然一行什么東西都沒有但是這一行確實(shí)存在,那么就不會(huì) fclose(fp); //遇到文件結(jié)束標(biāo)志。不僅換行不能交,空格也不能交 //即fprintf(fp,"%lf %lf %d ",s,s,w);、fprintf(fp,"%lf %lf %d ",s,s,w); 這兩種形式都錯(cuò) 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); //這個(gè)%d后面不能加'\n',因?yàn)樵谖募须m然一行什么東西都沒有但是這一行確實(shí)存在,那么就不會(huì) fclose(fp); //遇到文件結(jié)束標(biāo)志。不僅換行不能交,空格也不能交 //即fprintf(fp,"%lf %lf %d ",s,s,w);、fprintf(fp,"%lf %lf %d ",s,s,w); 這兩種形式都錯(cuò) 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); }
還有一種判斷文件結(jié)束方式:fgetc()
?但是這個(gè)函數(shù)相當(dāng)于getchar(),它會(huì)在文件中吸取一個(gè)字符,這樣的話文件指針就會(huì)向后移動(dòng)一位,從而導(dǎo)致拿出來的時(shí)候數(shù)據(jù)和進(jìn)去的時(shí)候不一樣
代碼:
#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); }
考慮到它判斷文件的方式,我們可以輸入的時(shí)候在每一條數(shù)據(jù)前面多加一個(gè)空格,來充當(dāng)那個(gè)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); //前面多加了一個(gè)空格。也可以加其他 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); }