C語(yǔ)言中炫酷的文件操作實(shí)例詳解
什么是文件
磁盤上的文件是文件
但是在程序設(shè)計(jì)中,我們一般談的文件有兩種:程序文件和數(shù)據(jù)文件(從文件功能的角度來(lái)分類)。
程序文件
包括源程序文件(例如.c文件)目標(biāo)文件(windows環(huán)境后綴為.obj)可執(zhí)行程序(windos環(huán)境后綴為exe)。
數(shù)據(jù)文件 (本文重點(diǎn))
文件的內(nèi)容不一定是程序,而是程序運(yùn)行時(shí)讀寫的數(shù)據(jù),比如程序運(yùn)行需要從中讀取數(shù)據(jù)的文件,或者輸出內(nèi)容的文件。
文件名
文件就像人一樣,他也要有姓氏和名字來(lái)讓其他文件或者人知道這個(gè)文件是誰(shuí)。
對(duì)于每一個(gè)文件要,,都有一個(gè)唯一的文件標(biāo)識(shí),以便用戶識(shí)別和引用。
文件名格式:文件路徑+文件名主干+文件后綴
例如:D:\CSDN\Test.txt
為了方便起見(jiàn),我們叫文件標(biāo)識(shí)為文件名
文件的打開(kāi)和關(guān)閉
文件指針
我們知道,指針是指向一個(gè)地址的,整形指針指向一個(gè)整形的空間,數(shù)組指針指向一個(gè)數(shù)組的空間,那么文件指針自然就是指向文件的指針了。
每個(gè)被使用的文件都在內(nèi)存中開(kāi)辟了一個(gè)相應(yīng)的文件信息區(qū),用來(lái)存放文件的相關(guān)信息(如文件的名字,文件狀態(tài)及文件當(dāng)前的位置等)。這些信息是保存在一個(gè)結(jié)構(gòu)體變量中的。該結(jié)構(gòu)體類型是有系統(tǒng)聲明的,取名FILE。
這是再vs的stdio頭文件下的文件信息區(qū)結(jié)構(gòu)體
struct _iobuf {
char *_ptr;
int _cnt;
char *_base;
int _flag;
int _file;
int _charbuf;
int _bufsiz;
char *_tmpfname;
};
typedef struct _iobuf FILE;
//不同的C編譯器的FILE類型包含的內(nèi)容不完全相同,但是大同小異。
就像學(xué)生要有學(xué)號(hào)姓名年齡這些信息一樣,文件也有他的信息,比如這個(gè)文件的地址。這些信息存放于這個(gè)結(jié)構(gòu)體中,通過(guò)typedef重命名為FILE,并且我們不需要關(guān)心一些細(xì)節(jié)(你會(huì)關(guān)心我昨天晚上吃了什么嗎)。
一般都是通過(guò)一個(gè)FILE的指針來(lái)維護(hù)這個(gè)FILE結(jié)構(gòu)的變量,這樣使用起來(lái)更加方便。
下面我們可以創(chuàng)建一個(gè)FILE*的指針變量:
FILE* pf;
pf通過(guò)該文件信息區(qū)中的信息就能夠訪問(wèn)該文件。
也就是說(shuō),通過(guò)文件指針變量能夠找到與它關(guān)聯(lián)的文件。
但上面的文件指針并未指向明確的位置,他暫時(shí)是一個(gè)野指針。
所以接下來(lái),我們來(lái)學(xué)習(xí)如何打開(kāi)(創(chuàng)建)一個(gè)文件。
文件函數(shù)
文件在讀寫之前應(yīng)該先打開(kāi)文件,在使用結(jié)束之后應(yīng)該關(guān)閉文件。
在編寫程序的時(shí)候,在打開(kāi)文件的同時(shí),都會(huì)返回一個(gè)FILE*的指針變量指向該文件,也相當(dāng)于建立了指針和文件的關(guān)系。
ANSIC規(guī)定使用fopen函數(shù)來(lái)打開(kāi)文件,fclose來(lái)關(guān)閉文件。
//打開(kāi)文件 FILE* fopen(const char *filename,const char *mode); //第一個(gè)參數(shù)是文件名,第二個(gè)參數(shù)是打開(kāi)方式 //關(guān)閉文件 int Fclose(FILE *stream);
部分打開(kāi)方式如下圖

現(xiàn)在我們來(lái)練習(xí)一下打開(kāi)文件
//打開(kāi)文件
FILE* pf = fopen("data.txt","r");
//以只讀的方式打開(kāi)這個(gè)文件
//如果文件打開(kāi)失敗會(huì)返回空,否則會(huì)返回指向該文件的指針
if (pf == NULL){
perror("fopen");
return -1;
}
//讀文件
//關(guān)閉文件
fclose(pf);
pf = NULL;
return 0;

但是卻打開(kāi)失敗了??!
原因是,從上面的打開(kāi)方式一圖我們可以看出,以"r"方式打開(kāi),需要該文件真實(shí)存在,但是我并沒(méi)有創(chuàng)建這個(gè)文件,所以打開(kāi)失敗了
叮~文件創(chuàng)建成功
我們?cè)賮?lái)看執(zhí)行結(jié)果

這回沒(méi)有報(bào)錯(cuò)
但是,這里是將data.txt文件放在了該.c文件目錄下,在我將該文件放在別的地方,仍然打開(kāi)失敗報(bào)錯(cuò)。
原因是,我們這段代碼只輸入了文件名,所以他只在當(dāng)前文件目錄下尋找該文件,在其他地方的文件我們就找不到了。
這里我們來(lái)看看兩個(gè)東西,一個(gè)叫相對(duì)路徑,一個(gè)叫絕對(duì)路徑
相對(duì)路徑與絕對(duì)路徑
相對(duì)路徑
只認(rèn)為是當(dāng)前目錄下的文件,如上面的代碼中。
絕對(duì)路徑
帶上文件的從磁盤到目標(biāo)文件的路徑
例如
D:\Program Files\data.txt
但是請(qǐng)注意,在編程中,\是轉(zhuǎn)義字符,所以我們需要讓\不再是轉(zhuǎn)義字符,使其代表它本身
D:\Program Files\data.txt
輸入輸出流
什么是輸入輸出流
學(xué)習(xí)過(guò)編程,一定知道printf或者cout或者System.out.println吧,
這些函數(shù)用于打印數(shù)據(jù),這就是標(biāo)準(zhǔn)的輸出流。使數(shù)據(jù)輸出或者寫入文件中,我們叫輸出流。
我們打印HELLO WORLD在屏幕上,就是一個(gè)標(biāo)準(zhǔn)輸出流
像scanf之類的,從文件輸入或者讀數(shù)據(jù)到內(nèi)存中,就是輸入流。
一些基本的輸入輸出函數(shù)如下。

比如,fputc就是寫一個(gè)字符進(jìn)去,fgetc就是讀一個(gè)字符。
//打開(kāi)文件
FILE* pf = fopen("data.txt","w");
if (pf == NULL){
perror("fopen");
return -1;
}
//讀文件
fputc('a', pf);
fputc('b', pf);
fputc('c', pf);
//fputc第一個(gè)參數(shù)為輸入的字符,第二個(gè)使對(duì)應(yīng)文件的指針
//關(guān)閉文件
fclose(pf);
pf = NULL;
return 0;

寫入了abc三個(gè)字符。
fputc和fgetc每次讀/寫一個(gè)字符后,文件指針pf會(huì)向后移動(dòng),類似strtok函數(shù)。會(huì)記錄上一次輸入/輸出的地址。
如果不這樣,那豈不是一直在一個(gè)位置重復(fù)寫入或者讀文件了。
接下來(lái)看看fgetc讀取字符
//打開(kāi)文件
FILE* pf = fopen("data.txt","r");
if (pf == NULL){
perror("fopen");
return -1;
}
//讀文件
int a = fgetc(pf);
printf("%c", a);
a = fgetc(pf);
printf("%c", a);
a = fgetc(pf);
printf("%c",a);
//關(guān)閉文件
fclose(pf);
pf = NULL;
return 0;
執(zhí)行結(jié)果

這就是順序讀寫,按著順序讀入寫入。
當(dāng)然,有順序讀寫,就會(huì)有隨機(jī)讀寫
從字面意思就能看到,隨機(jī)讀寫emmm。
當(dāng)然,除了fgetc這類,fgets自然就是讀取一行了(只會(huì)讀/寫一行哦)
如果你用這類函數(shù)輸出在標(biāo)準(zhǔn)輸入或者標(biāo)準(zhǔn)輸入(stdout或者stdin)上,他和printf,scanf沒(méi)什么區(qū)別。
接下來(lái),我們來(lái)看二進(jìn)制的讀和寫
二進(jìn)制讀寫
fwirte

以二進(jìn)制的形式將內(nèi)容寫入文件中
第一個(gè)參數(shù)是你要寫入數(shù)據(jù)的數(shù)據(jù)地址,第二個(gè)參數(shù)是一個(gè)類型的大?。ㄗ止?jié))。第三個(gè)參數(shù)是你要寫入幾個(gè)數(shù)據(jù),第四個(gè)則是你選定寫入的流。
struct S{
int n;
double d;
char name[10];
};
int main()
{
struct S s = { 10, 3.14, "zhangsan" };
//打開(kāi)文件
FILE* pf = fopen("data.txt","wb");
if (pf == NULL){
perror("fopen");
return -1;
}
//讀文件
fwrite(&s,sizeof(s),1,pf);
//關(guān)閉文件
fclose(pf);
pf = NULL;
return 0;
}
如上代碼會(huì)將數(shù)據(jù)以二進(jìn)制的形式寫入data.txt
雖然我們看不懂,但是能看到zhangsan是我們輸入的內(nèi)容

fread
以二進(jìn)制的形式讀

和fwirte一樣,只不過(guò)buffer不是const形式了,因?yàn)槲覀円獙?shù)據(jù)讀入該指針指向的目標(biāo)。
struct S{
int n;
double d;
char name[10];
};
int main()
{
struct S s = {0};
//打開(kāi)文件
FILE* pf = fopen("data.txt","rb");
if (pf == NULL){
perror("fopen");
return -1;
}
//讀文件
fread(&s,sizeof(struct S),1,pf);
printf("%d %lf %s\n",s.n,s.d,s.name);
//關(guān)閉文件
fclose(pf);
pf = NULL;
return 0;
}

總結(jié)
到此這篇關(guān)于C語(yǔ)言中文件操作的文章就介紹到這了,更多相關(guān)C語(yǔ)言文件操作內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
OpenCV中findContours函數(shù)參數(shù)詳解
Opencv中通過(guò)使用findContours函數(shù),簡(jiǎn)單幾個(gè)的步驟就可以檢測(cè)出物體的輪廓,很方便。本文將和大家一起探討一下findContours方法中各參數(shù)的含義及用法,感興趣的可以了解一下2022-08-08
深入解析C++編程中__alignof 與__uuidof運(yùn)算符的使用
這篇文章主要介紹了C++編程中__alignof 與__uuidof運(yùn)算符的使用,是C++入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2016-01-01
C語(yǔ)言運(yùn)用回調(diào)函數(shù)實(shí)現(xiàn)計(jì)算器
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言運(yùn)用回調(diào)函數(shù)實(shí)現(xiàn)計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10
數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)-用棧實(shí)現(xiàn)表達(dá)式求值的方法詳解
本篇文章是對(duì)在c語(yǔ)言中用棧實(shí)現(xiàn)表達(dá)式求值的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05

