用C語言如何實(shí)現(xiàn)文件的讀寫功能?
讀寫文件在編程里面應(yīng)該是用的最多也最基礎(chǔ)的工作了。但是,C/C++中文件讀寫方法有很多種,各種方法的使用場(chǎng)景區(qū)別甚大,到底日常工作中,我們什么時(shí)候該用哪種方法呢?還是說隨便抓一種用就可以?
這篇文章就給大家分享一下C語言文件的各種方法,下一篇再分享一下C++的讀寫方法:
在window環(huán)境下,文件分別兩種,一種為文本文件(用記事本可以打開的文件,如txt、xml、html、js等),一種為二進(jìn)制文件(如圖片、軟件、壓縮包等)
讀寫文本文件首先創(chuàng)建一個(gè)File對(duì)象,這個(gè)利用到系統(tǒng)庫中的fopen()函數(shù),該函數(shù)返回一個(gè)File*類型的對(duì)象。
FILE *fopen( const char * filename, const char * mode );
第一個(gè)參數(shù)為:文件路徑;第二個(gè)參數(shù)為打開模式:這個(gè)是固定的,具體是什么意思大家可以自己去查一下,沒什么特別的,記住就好了。
1、文本文件讀寫
接下來我們讀寫一個(gè)文本文件,我新建一個(gè)test.txt文件并且讀取出內(nèi)容寫進(jìn)newtest.txt文件中,這里有兩組函數(shù)fgetc/fputc和fgets/fputs函數(shù),故名意意,fgetc是讀取出一個(gè)字符,而fgets是讀取一段字符串,fputc是讀寫入一個(gè)字符,fputs寫進(jìn)一段字符串。我們分別來看一下:
int main(){ FILE * fpr = fopen("test.txt","r"); FILE * fpw = fopen("newtest.txt","w"); if(fpr && fpw) { while (!feof(fpr)) { char c = (char)fgetc(fpr); cout<<c; fputc(c,fpw); } } fclose(fpr); fclose(fpw); system("pause"); return 0 ;}注意:
1、feof是判斷是否讀取到文件尾的函數(shù),此處不用EOF來作為讀寫結(jié)束條件是因?yàn)椋汉瘮?shù)如fgetc或getc返回EOF并不一定表示文件結(jié)束,當(dāng)讀取文件出錯(cuò)時(shí)也會(huì)返回EOF,僅憑返回-1就認(rèn)為文件結(jié)束是錯(cuò)誤的;正因?yàn)槿绱耍覀冃枰猣eof()來判斷文件是否結(jié)束,當(dāng)然用feof()來判斷文件結(jié)束時(shí)也需要判斷讀取操作是否出錯(cuò),這時(shí)可以用ferror()來判斷,當(dāng)其為真時(shí)表示有錯(cuò)誤發(fā)生。在實(shí)際的程序中,應(yīng)該每執(zhí)行一次文件操作,就用用ferror函數(shù)檢測(cè)是否出錯(cuò)。
2、fgetc()返回的是int,如果要在屏幕上打印出字符的話需要轉(zhuǎn)換為char;
上面是一個(gè)字符一個(gè)字符的讀,下面一段一段的讀:
FILE * fpr = fopen("test.txt","r"); FILE * fpw = fopen("newtest.txt","w"); if(fpr && fpw) { while (!feof(fpr)) { char* buff = new char[1024]; char* s = fgets(buff,3,fpr); cout<<s; fputs(s,fpw); } } fclose(fpr); fclose(fpw);這個(gè)方法個(gè)人不是很推薦,因?yàn)樵谑褂眠^程中經(jīng)常會(huì)報(bào)錯(cuò),這里我如果將以此讀取的字符串長(zhǎng)度由3改為2或者1就會(huì)報(bào)錯(cuò),這個(gè)我也搞不清為什么,如果有知道的朋友可以評(píng)論幫我解答一下;
讀寫二進(jìn)制文件C語言讀寫二進(jìn)制文件通過庫函數(shù)fread和fwrite實(shí)現(xiàn):
size_t fread(void *buff, size_t sizeofElement, size_t count, FILE *stream);
size_t fwrite(const void *ptr, size_t sizeofElement, size_t count, FILE *stream);
參數(shù)解釋:buff:緩存區(qū)地址
sizeofElement:每次讀/寫的數(shù)據(jù)大小
count:要多去多個(gè)個(gè)sizeofElement大小的數(shù)據(jù)
stream:文件指針
現(xiàn)在我們來讀一張神仙姐姐的圖片test.ipg并且將讀取到的內(nèi)容寫進(jìn)newtest.jpg,然后查看是否能打開。
int main(){FILE * fpr = fopen("test.jpg","rb");FILE * fpw = fopen("newtest.jpg","wb");char* buff = new char[50];int len = 50;while (len==50){len = fread(buff,sizeof(char),50,fpr);fwrite(buff,sizeof(char),len,fpw);}fclose(fpr);fclose(fpw);system("pause");return 0 ;}運(yùn)行結(jié)果,打開newtest.jpg,結(jié)果完美呈現(xiàn)神仙姐姐。
注意以下幾點(diǎn):
1、文件打開方式,不管是讀還是寫都需要在后面加b,b就是binary二進(jìn)制的意思,所以這里打開方式是“日本”和“wb”;
2、寫入fwrite中的count一定要注意,你的緩存區(qū)存了多大的內(nèi)容這里就寫多大的內(nèi)容,不然會(huì)導(dǎo)致文件損壞,因?yàn)槲叶ǖ氖敲看巫x取50個(gè)char大小的內(nèi)容,但是文件大小不可能是50的整數(shù)啊,最后一次讀取的話他肯定會(huì)小于50,并且大小不能超過緩存區(qū)的大小;
文件內(nèi)部位置指針
在打開任何文件的時(shí)候,都會(huì)有一個(gè)文件內(nèi)部位置指針。為什么要在這里介紹這個(gè)呢?我們大部分讀取文件的時(shí)候都是不用去管他的,因?yàn)槲覀兓竞苌儆凶x取固定位置開始固定長(zhǎng)度的需求,都是從開始一直讀取到文件結(jié)束。
其實(shí)我們上面用到的那么多函數(shù),位置指針在文件被打開的時(shí)候位置指針是在零為,每讀取一次這個(gè)指針就會(huì)往后面移動(dòng)你讀取的數(shù)據(jù)兩個(gè)距離。
這里有三個(gè)函數(shù):
1、int fseek(FILE *stream, long int offset, int where)這個(gè)函數(shù)的作用是將文件位置指針從指定位置(where:系統(tǒng)有定義三個(gè)宏SEEK_SET:文件頭、SEEK_CUR當(dāng)前指針位置、SEEK_END文件尾,當(dāng)然你也可以自己指定位置,比如100、50這樣)移動(dòng)個(gè)offset個(gè)位置(正數(shù)往后移,負(fù)數(shù)往前移);
2、void rewind(FILE *stream)這個(gè)很簡(jiǎn)單,將位置指針指向文件開頭;
3、long int ftell(FILE *stream)這個(gè)也很簡(jiǎn)答,告訴你當(dāng)前位置指針的位置(相對(duì)于文件頭);
這可以干嘛呢?最實(shí)用的,當(dāng)然是快速的獲取文件長(zhǎng)度啊:
我們只需要用fseek將指針移動(dòng)到文件尾,再用ftell求出當(dāng)前指針位置就是了:
fseek(fpr,OL,SEEK_END);注意這里的offset一定要給0
ftell(fpr);