用C語言如何實現文件的讀寫功能?
讀寫文件在編程里面應該是用的最多也最基礎的工作了。但是,C/C++中文件讀寫方法有很多種,各種方法的使用場景區別甚大,到底日常工作中,我們什么時候該用哪種方法呢?還是說隨便抓一種用就可以?
這篇文章就給大家分享一下C語言文件的各種方法,下一篇再分享一下C++的讀寫方法:
在window環境下,文件分別兩種,一種為文本文件(用記事本可以打開的文件,如txt、xml、html、js等),一種為二進制文件(如圖片、軟件、壓縮包等)
讀寫文本文件首先創建一個File對象,這個利用到系統庫中的fopen()函數,該函數返回一個File*類型的對象。
FILE *fopen( const char * filename, const char * mode );
第一個參數為:文件路徑;第二個參數為打開模式:這個是固定的,具體是什么意思大家可以自己去查一下,沒什么特別的,記住就好了。
1、文本文件讀寫
接下來我們讀寫一個文本文件,我新建一個test.txt文件并且讀取出內容寫進newtest.txt文件中,這里有兩組函數fgetc/fputc和fgets/fputs函數,故名意意,fgetc是讀取出一個字符,而fgets是讀取一段字符串,fputc是讀寫入一個字符,fputs寫進一段字符串。我們分別來看一下:
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是判斷是否讀取到文件尾的函數,此處不用EOF來作為讀寫結束條件是因為:函數如fgetc或getc返回EOF并不一定表示文件結束,當讀取文件出錯時也會返回EOF,僅憑返回-1就認為文件結束是錯誤的;正因為如此,我們需要feof()來判斷文件是否結束,當然用feof()來判斷文件結束時也需要判斷讀取操作是否出錯,這時可以用ferror()來判斷,當其為真時表示有錯誤發生。在實際的程序中,應該每執行一次文件操作,就用用ferror函數檢測是否出錯。
2、fgetc()返回的是int,如果要在屏幕上打印出字符的話需要轉換為char;
上面是一個字符一個字符的讀,下面一段一段的讀:
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);這個方法個人不是很推薦,因為在使用過程中經常會報錯,這里我如果將以此讀取的字符串長度由3改為2或者1就會報錯,這個我也搞不清為什么,如果有知道的朋友可以評論幫我解答一下;
讀寫二進制文件C語言讀寫二進制文件通過庫函數fread和fwrite實現:
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);
參數解釋:buff:緩存區地址
sizeofElement:每次讀/寫的數據大小
count:要多去多個個sizeofElement大小的數據
stream:文件指針
現在我們來讀一張神仙姐姐的圖片test.ipg并且將讀取到的內容寫進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 ;}運行結果,打開newtest.jpg,結果完美呈現神仙姐姐。
注意以下幾點:
1、文件打開方式,不管是讀還是寫都需要在后面加b,b就是binary二進制的意思,所以這里打開方式是“日本”和“wb”;
2、寫入fwrite中的count一定要注意,你的緩存區存了多大的內容這里就寫多大的內容,不然會導致文件損壞,因為我定的是每次讀取50個char大小的內容,但是文件大小不可能是50的整數啊,最后一次讀取的話他肯定會小于50,并且大小不能超過緩存區的大小;
文件內部位置指針
在打開任何文件的時候,都會有一個文件內部位置指針。為什么要在這里介紹這個呢?我們大部分讀取文件的時候都是不用去管他的,因為我們基本很少有讀取固定位置開始固定長度的需求,都是從開始一直讀取到文件結束。
其實我們上面用到的那么多函數,位置指針在文件被打開的時候位置指針是在零為,每讀取一次這個指針就會往后面移動你讀取的數據兩個距離。
這里有三個函數:
1、int fseek(FILE *stream, long int offset, int where)這個函數的作用是將文件位置指針從指定位置(where:系統有定義三個宏SEEK_SET:文件頭、SEEK_CUR當前指針位置、SEEK_END文件尾,當然你也可以自己指定位置,比如100、50這樣)移動個offset個位置(正數往后移,負數往前移);
2、void rewind(FILE *stream)這個很簡單,將位置指針指向文件開頭;
3、long int ftell(FILE *stream)這個也很簡答,告訴你當前位置指針的位置(相對于文件頭);
這可以干嘛呢?最實用的,當然是快速的獲取文件長度啊:
我們只需要用fseek將指針移動到文件尾,再用ftell求出當前指針位置就是了:
fseek(fpr,OL,SEEK_END);注意這里的offset一定要給0
ftell(fpr);