溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何進行C語言文件操作

發(fā)布時間:2021-10-09 11:24:46 來源:億速云 閱讀:121 作者:柒染 欄目:開發(fā)技術(shù)

如何進行C語言文件操作,相信很多沒有經(jīng)驗的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。

    一、前言

    我們?nèi)绾问刮覀冊O(shè)計的程序具有“記憶功能”呢?答案是將數(shù)據(jù)以文件的形式另外保存。保存的形式有很多,在本文中我們以最簡單的文本形式保存在記事本上,相信這篇文章一定讓你學會。

    二、文件操作基礎(chǔ)知識

    ①什么是文件

    我們一般談的文件有兩種:程序文件、數(shù)據(jù)文件
    <程序文件>
    包括源程序文件(后綴為.c),目標文件(windows環(huán)境后綴為.obj),可執(zhí)行程序(windows環(huán)境后綴為.exe)。
    <數(shù)據(jù)文件>
    文件的內(nèi)容不一定是程序,而是程序運行時讀寫的數(shù)據(jù),比如程序運行需要從中讀取數(shù)據(jù)的文件,或者輸出內(nèi)容的文件。

    本章討論的是數(shù)據(jù)文件

    ②數(shù)據(jù)文件類型

    文本文件與二進制文件

    數(shù)據(jù)在內(nèi)存中以二進制的形式存儲,如果不加轉(zhuǎn)換的輸出到外存,就是<二進制文件>(后綴.bin)。

    如果要求在外存上以ASCII碼的形式存儲,則需要在存儲前轉(zhuǎn)換。以ASCII字符的形式存儲的文件就是<文本文件>(后綴.txt)。

    ③數(shù)據(jù)如何存儲

    一個數(shù)據(jù)在內(nèi)存中是怎么存儲的呢?
    字符一律以ASCII形式存儲,數(shù)值型數(shù)據(jù)既可以用ASCII形式存儲,也可以使用二進制形式存儲。
    如有整數(shù)10000,如果以ASCII碼的形式輸出到磁盤,則磁盤中占用5個字節(jié)(按照一個字符一個字符儲存,1  0  0  0  0每個字符一個字節(jié)),而二進制形式輸出,則在磁盤上只占4個字節(jié)(int)

    ④如何讀取二進制文件

    文本文件是我們通過記事本可以直接理解讀取的,而二進制文件如果以記事本的形式打開是我們不能識別的亂碼。但vs可以以某種方式讀取二進制文件,方法如下:

    (下圖表示以記事本讀取以二進制形式保存的10000) 

    如何進行C語言文件操作

     演示二進制文件打開方式: 

     第一步:將文件添加到vs下

    如何進行C語言文件操作

     第二步:右擊后打開方式選擇二進制編譯器

    如何進行C語言文件操作

     這就是最后的效果了。

    如何進行C語言文件操作

    ⑤什么是文件名

    文件名包含3部分:文件路徑+文件名主干+文件后綴
    例如: c: \code\ test.txt  (在后文會提到對文件名的絕對引用和相對引用)

    ⑥文件緩沖區(qū)

    如何進行C語言文件操作

    含義:ANSIC 標準采用“緩沖文件系統(tǒng)”處理的數(shù)據(jù)文件。所謂緩沖文件系統(tǒng)是指系統(tǒng)自動地在內(nèi)存中為程序中每一個正在使用的文件開辟一塊“文件緩沖區(qū)”。文件緩沖區(qū)是用以暫時存放讀寫期間的文件數(shù)據(jù)而在內(nèi)存區(qū)預留的一定空間。通過磁盤緩存來實現(xiàn),磁盤緩存本身并不是一種實際存在的存儲介質(zhì),它依托于固定磁盤,提供對主存儲器存儲空間的擴充,即利用主存中的存儲空間, 來暫存從磁盤中讀出(或?qū)懭?的信息。。

    特點:從內(nèi)存向磁盤輸出數(shù)據(jù)會先送到內(nèi)存中的緩沖區(qū),裝滿緩沖區(qū)后才一起送到磁盤上。如果從磁盤向計算機讀入數(shù)據(jù),則從磁盤文件中讀取數(shù)據(jù)輸入到內(nèi)存緩沖區(qū)(充滿緩沖區(qū)),然后再從緩沖區(qū)逐個地將數(shù)據(jù)送到程序數(shù)據(jù)區(qū)(程序變量等)。緩沖區(qū)的大小根據(jù)C編譯系統(tǒng)決定的 。

    [擴展-三種類型的緩存區(qū)]       鏈接-文件緩沖區(qū)

    ⑦文件指針

    <文件指針>

    緩沖文件系統(tǒng)中,關(guān)鍵的概念是“文件類型指針”,簡稱“文件指針”。每個被使用的文件都在內(nèi)存中開辟了一個相應的文件信息區(qū),用來存放文件的相關(guān)信息(如文件的名字,文件狀態(tài)及
    文件當前的位置等)。這些信息是保存在一個結(jié)構(gòu)體變量中的。該結(jié)構(gòu)體類型是有系統(tǒng)聲明的,取名FILE。

    每當打開一個文件的時候,系統(tǒng)會根據(jù)文件的情況自動創(chuàng)建一個FILE結(jié)構(gòu)的變量,并填充其中的信息,使用者不必關(guān)心細節(jié)。一般都是通過一個FILE的指針來維護這個FILE結(jié)構(gòu)的變量,這樣使用起來更加方便。
    FILE*p//文件類型指針

    可以使pf指向某個文件的文件信息區(qū)(是一個結(jié)構(gòu)體變量)。通過該文件信息區(qū)中的信息就能夠訪問該文件。也就是說,通過文件指針變量能夠找到與它關(guān)聯(lián)的文件。

    如何進行C語言文件操作

    三、文件操作函數(shù)

    ①fopen 與 fclose

    1.fopen

    如何進行C語言文件操作

    <功能>打開文件

    <參數(shù)>  filename-文件名         mode-打開方式        

    <返回值>一個文件指針

    (打開方式mode表)

    如何進行C語言文件操作

     2.fclose

    如何進行C語言文件操作

    <功能>  關(guān)閉文件

    <參數(shù)>  stream-文件指針 

    3.使用示范

    #include<stdio.h>
    #include<sting.h>
    #include<errno.h>
    int main()
    {
    	FILE*pw = fopen("test.txt","w");//打開的文件為“test.txt”      打開方式為“w”
        if(pw==NULL)//若打開失敗則說明失敗原因并結(jié)束進程
        {
            printf("%s",strerror(errno));//errno是全局錯誤變量  strerror將errno解析為錯誤原因
            return 0;
        }
    	fclose(pw);//關(guān)閉文件
    	pw = NULL;//將指針置為空,防止被誤用
    	return 0;
    }

    文件名相對/絕對路徑

    絕對路徑:如c:\code\test.txt     包含文件路徑  文件主干  文件后綴

    相對路徑:上圖的表示就是相對路徑,表示txt文件與源文件在同一路徑下,若要表示上一路                    勁,則用“../”,一次類推上上路徑就是“../../”

    FILE*pw = fopen("../test.txt","w");

    ②fputc與fgetc

    1.fputc——輸出函數(shù)

    如何進行C語言文件操作

    <功能>將一個字符寫入流中

    <參數(shù)>c-輸出的字符

    <返回值>正?!祷剌敵龅淖址?/p>

                   錯誤——返回EOF

    <適用>所有流

    2.fgetc——輸入函數(shù)

    如何進行C語言文件操作

    <功能>從流中讀取一個字符

    <返回值>int——返回輸入的字符

                   EOF——發(fā)生錯誤或到達文件結(jié)尾

    <適用>所有流 

     3.使用示范

    #include <stdio.h>
    #include <string.h>
    #include<errno.h>
    //fputc   輸出一個字母
    int main()
    {
    	FILE*pw = fopen("test.txt","w");
    	if (pw == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	fputc('b',pw);//將字符‘b'寫入文件"test.txt"
    	fclose(pw);
    	pw = NULL;
    	return 0;
    }
     
    //fgetc 讀取一個字符,讀取一個后文件指針往后偏移一位
    int main()
    {
    	FILE*pr = fopen("test.txt","r");
    	if (pr == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	char ch = 0;//用ch接收輸入的字符
    	ch = fgetc(pr);//從文件"test.txt"中讀取,讀取一個字符后文件指針自動往后移一位
    	fclose(pr);
    	pr = NULL;
    	printf("%c",ch);
    	return 0;
    }

     4.對所有流的理解

    什么是流:流是指信息從外部輸入設(shè)備(如鍵盤)向計算機內(nèi)部(如內(nèi)存)輸入和從內(nèi)存                        向外部輸出設(shè)備(顯示器)輸出的過程。                       

    上述代碼演示的是對文件流的操作,我們在用標準輸入(stdin)輸出(stdout)流演示一下

    如何進行C語言文件操作

    ③fputs與fgets

    1.fputs——輸出函數(shù)

    如何進行C語言文件操作

    <功能>將字符串寫入流中

    <返回值>int——非負值表示成功

                   EOF——發(fā)生錯誤

    <適用>所有流

    2.fgets——輸入函數(shù)

    如何進行C語言文件操作

    <功能>從流中獲取字符串

    <返回值>正?!祷刈址?/p>

                   NULL——表示錯誤或者到達文件結(jié)尾

    <參數(shù)>n-從流中讀取的最大字符數(shù)(\0會自動占去一位)

    3.使用示范

    #include <stdio.h>
    #include <string.h>
    #include<errno.h>
    //fputs
    int main()
    {
    	FILE*pw = fopen("test.txt","w");
    	if (pw == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	char ch[5] = "abcd";
    	fputs(ch,pw);
    	fclose(pw);
    	pw = NULL;
    	return 0;
    }
     
    //fgets
    int main()
    {
    	FILE*pr = fopen("test.txt","r");
    	if (pr == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	char ch[5] = {0};
    	fgets(ch,3,pr);//有一位自動被\0占用
    	printf("%s",ch);
    	return 0;
    }

    ④fprintf與fscanf

    1.fprintf——輸出函數(shù)

    如何進行C語言文件操作

    如何進行C語言文件操作

    <功能>將特定格式的數(shù)據(jù)寫入流中

    <返回值>返回打印的字節(jié)數(shù)。

    <適用>所有流

    <對比>printf默認將數(shù)據(jù)打印在標準輸出流(stdout)上,而fprintf的輸出流可以選擇。printf             打印的字符數(shù)返回

     2.fscanf——輸入函數(shù)

    如何進行C語言文件操作

    如何進行C語言文件操作

    <功能>從流中讀取特定格式的數(shù)據(jù)

    <適用>所有流

    <返回值>返回成功轉(zhuǎn)換和分配的字段數(shù)量

    3.使用示范

    //頭文件同上略
    //fprintf
    int main()
    {
    	FILE*pw = fopen("test.txt","w");
    	if (pw == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	fprintf(pw,"%d %.2f %c",10,3.14,'a');
    	fclose(pw);
    	pw = NULL;
    	return 0;
    }
     
    //fscanf
    int main()
    {
    	int a; float b; char c;
    	FILE*pr = fopen("test.txt","r");
    	if (pr == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	fscanf(pr,"%d %f %c",&a,&b,&c);
    	printf("%d %.2f %c", a,b,c);
    	return 0;
    }

    ⑤fwrite與fread

    1.fwrite——輸出函數(shù)

    如何進行C語言文件操作

    <功能>將數(shù)據(jù)以二進制的形式寫入文件流中

    <參數(shù)>buffer-指向?qū)懭霐?shù)據(jù)的指針    size-每一個元素的大小   count-寫入的最大元素數(shù)

    <返回值>實際寫入的元素數(shù)

    <適用>文件流

    2.fread——輸入函數(shù)

    如何進行C語言文件操作

    <功能>將數(shù)據(jù)以二進制的形式從文件流中讀取

    <返回值>實際讀入的元素數(shù)

    <適用>文件流

     3.使用示范

    //頭文件同上略
    //fwrite  fread
    typedef struct stu
    {
    	int n;
    	float score;
    	char name[10];
    }stu;
     
    int main()
    {
    	stu s = {10,100.0,"張三"};
    	FILE*pr = fopen("test.txt","rb");
    	struct stu s1 = { 0 };
    	if (pr == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	//fwrite(&s,sizeof(stu),1,pw);
    	fread(&s1,sizeof(stu),1,pr);
    	fclose(pr);
    	pr = NULL;
    	return 0;
    }

    ⑥fseek與ftell與rewind

    1.fseek

    如何進行C語言文件操作

    <功能>將文件指針移到指定的位置

    <參數(shù)>offset-偏移量        origin-初始化文件指針位置

                origin有三種選擇:SEEK_CUR  從當前位置開始

                                               SEEK_END  從文件結(jié)尾開始

                                               SEEK_SET   從文件開頭開始

    <返回值>成功返回0,失敗返回非0值

    <注意>對文件指針的修改不可以通過p++的方式實先

     使用演示(現(xiàn)已知記事本儲存數(shù)據(jù)為“abcdef”)

    //fseek
    int main()
    {
    	FILE*pr = fopen("test.txt","r");
    	int a = 0;
    	if (pr == NULL)
    	{
    		printf("%s",strerror(errno));
    		return 0;
    	}
    	//定位文件指針
    	fseek(pr, -1, SEEK_END);//指向最后一個的后一個
    	a=fgetc(pr);
        b=ftell(pr);//返回當前指針位置
    	printf("%c",a);
        printf("%d",b);
    	fclose(pr);
    	pr = NULL;
    	return 0;
    }

    (若初始化文件指針為SEEK_SET,偏移1則得到b)

    2.ftell

    如何進行C語言文件操作

    <功能>返回當前文件指針的偏移量 

    3.rewind

    如何進行C語言文件操作

    <功能>使文件指針回到文件的開始

    使用演示

    int main()
    {
    	int n;
    	FILE * pFile;
    	char buffer[27];
    	pFile = fopen("test.txt", "w+");
    	for (n = 'A'; n <= 'Z'; n++)
    		fputc(n, pFile);
    	rewind(pFile);
    	fread(buffer, 1, 26, pFile);
    	fclose(pFile);
    	buffer[26] = '\0';
    	puts(buffer);
    	return 0;
    }

    ⑦ferror與feof

    1.ferror

    如何進行C語言文件操作

    <功能>檢查流是否發(fā)生了錯誤

    <返回值>若發(fā)生錯誤則返回0,否則返回非0值

    2.feof

    如何進行C語言文件操作  

    <功能>當文件讀取結(jié)束的時候,判斷是讀取失敗結(jié)束,還是遇到文件尾結(jié)束

    <返回值>若不是文件末尾則返回0,是文件末尾則返回非0值

    <注意>不能用feof函數(shù)的返回值直接用來判斷文件的是否結(jié)束

    1. 文本文件讀取是否結(jié)束,判斷返回值是否為EOF (fgetc),或者NULL(fgets)
    例如:
    fgetc判斷是否為EOF.
    fgets判斷返回值是否為NULL.
    2. 二進制文件的讀取結(jié)束判斷,判斷返回值是否小于實際要讀的個數(shù)。
    例如:
    fread判斷返回值是否小于實際要讀的個數(shù)

    使用演示(對于文本文件)

    int main(void)
    {
    	int c; // 注意:int,非char,要求處理EOF,EOF本質(zhì)上是-1
    	FILE* fp = fopen("test.txt", "r");
    	if (!fp)
        {
    		perror("File opening failed");
    		return EXIT_FAILURE;
    	}
    	//fgetc 當讀取失敗的時候或者遇到文件結(jié)束的時候,都會返回EOF
    	while ((c = fgetc(fp)) != EOF) // 標準C I/O讀取文件循環(huán)
    	{
    		putchar(c);
    	}
    	//判斷是什么原因結(jié)束的
    	if (ferror(fp))
    		puts("I/O error when reading");
    	else if (feof(fp))
    		puts("End of file reached successfully");
    	fclose(fp);
    }

    使用演示(對于二進制文件)

    enum { SIZE = 5 };
    int main(void)
    {
    	double a[SIZE] = { 1.0, 2.0, 3.0, 4.0, 5.0 };
    	double b = 0.0;
    	size_t ret_code = 0;
    	FILE *fp = fopen("test.bin", "wb"); // 必須用二進制模式
    	fwrite(a, sizeof(*a), SIZE, fp); // 寫 double 的數(shù)組     sizeof(*)表示這種類型大小
    	fclose(fp);
    	fp = fopen("test.bin", "rb");
    	// 讀 double 的數(shù)組
    	while ((ret_code = fread(&b, sizeof(double), 1, fp)) >= 1)
    	{
    		printf("%lf\n", b);
    	}
    	if (feof(fp))
    		printf("Error reading test.bin: unexpected end of file\n");
    	else if (ferror(fp)) {
    		perror("Error reading test.bin");
    	}
    	fclose(fp);
    	fp = NULL;
    }

    ⑧補充函數(shù) sscanf sprintf  

    1.sprintf

    如何進行C語言文件操作

    <功能>將特定格式的數(shù)據(jù)寫入字符串

    <參數(shù)>buffer-輸出的儲存位置 

    <返回值>寫入字符串的數(shù)據(jù)大?。▎挝蛔止?jié))

    2.sscanf

    如何進行C語言文件操作

    <功能>從字符串中讀取特定格式的字符串

    3.使用演示

    //sprintf  sscanf
    typedef struct stu
    {
    	int n;
    	float score;
    	char name[10];
    }stu;
     
    int main()
    {
    	stu s = {10,3.14,"張三"};
    	char buf1[1024] = {0};
    	stu s1 = {0};
    	sprintf(buf1, "%d %f %s",s.n,s.score,s.name);//不要再加上n=...,score=...
    	sscanf(buf1,"%d %f %s",&(s1.n),&(s1.score),&(s1.name));
    	printf("%d\n",s1.n);
    	printf("%s\n",s1.name);
    	return 0 ;
    }

    ⑨補充函數(shù)perror  strerror

    如何進行C語言文件操作

    如何進行C語言文件操作

    如何進行C語言文件操作

     <比較>perror和strerror相比不需要引用errno變量,也不需要使用printf函數(shù),操作更加便捷。同時自行輸入的字符串起到標識的作用,不會混淆。

    總結(jié)

    到此這篇關(guān)于C語言文件操作零基礎(chǔ)新手入門保姆級教程的文章就介紹到這了,更多相關(guān)C語言文件操作零基礎(chǔ)教程內(nèi)容請搜索億速云以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持億速云!

    看完上述內(nèi)容,你們掌握如何進行C語言文件操作的方法了嗎?如果還想學到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

    向AI問一下細節(jié)

    免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

    AI