文件讀寫
一個文件羹唠,無論它是文本文件還是二進(jìn)制文件凸椿,都是代表了一系列的字節(jié)际起。C 語言不僅提供了訪問頂層的函數(shù)悍缠,也提供了底層(OS)調(diào)用來處理存儲設(shè)備上的文件。
打開文件
您可以使用 fopen( ) 函數(shù)來創(chuàng)建一個新的文件或者打開一個已有的文件姑荷,這個調(diào)用會初始化類型 FILE 的一個對象盒延,類型 FILE 包含了所有用來控制流的必要的信息缩擂。下面是這個函數(shù)調(diào)用的原型:
FILE *fopen( const char * filename, const char * mode );
在這里,filename 是字符串添寺,用來命名文件胯盯,訪問模式 mode 的值可以是下列值中的一個:
模式 | 描述 |
---|---|
r | 打開一個已有的文本文件,允許讀取文件计露。 |
w | 打開一個文本文件陨闹,允許寫入文件。如果文件不存在薄坏,則會創(chuàng)建一個新文件。在這里寨闹,您的程序會從文件的開頭寫入內(nèi)容胶坠。 |
a | 打開一個文本文件,以追加模式寫入文件繁堡。如果文件不存在沈善,則會創(chuàng)建一個新文件。在這里椭蹄,您的程序會在已有的文件內(nèi)容中追加內(nèi)容闻牡。 |
r+ | 打開一個文本文件,允許讀寫文件绳矩。 |
w+ | 打開一個文本文件罩润,允許讀寫文件。如果文件已存在翼馆,則文件會被截斷為零長度割以,如果文件不存在,則會創(chuàng)建一個新文件应媚。 |
a+ | 打開一個文本文件严沥,允許讀寫文件。如果文件不存在中姜,則會創(chuàng)建一個新文件消玄。讀取會從文件的開頭開始,寫入則只能是追加模式丢胚。 |
如果處理的是二進(jìn)制文件翩瓜,則需使用下面的訪問模式來取代上面的訪問模式:
"rb", "wb", "ab", "rb+", "r+b", "wb+", "w+b", "ab+", "a+b"
關(guān)閉文件
為了關(guān)閉文件,請使用 fclose( ) 函數(shù)嗜桌。函數(shù)的原型如下:
int fclose( FILE *fp );
如果成功關(guān)閉文件奥溺,fclose( ) 函數(shù)返回零,如果關(guān)閉文件時發(fā)生錯誤骨宠,函數(shù)返回 EOF浮定。這個函數(shù)實際上相满,會清空緩沖區(qū)中的數(shù)據(jù),關(guān)閉文件桦卒,并釋放用于該文件的所有內(nèi)存立美。EOF 是一個定義在頭文件 stdio.h 中的常量。
C 標(biāo)準(zhǔn)庫提供了各種函數(shù)來按字符或者以固定長度字符串的形式讀寫文件方灾。
讀取文件
下面是從文件讀取單個字符的最簡單的函數(shù):
int fgetc( FILE * fp );
fgetc() 函數(shù)從 fp 所指向的輸入文件中讀取一個字符建蹄。返回值是讀取的字符,如果發(fā)生錯誤則返回 EOF裕偿。下面的函數(shù)允許您從流中讀取一個字符串:
char *fgets( char *buf, int n, FILE *fp );
函數(shù) fgets() 從 fp 所指向的輸入流中讀取 n - 1 個字符洞慎。它會把讀取的字符串復(fù)制到緩沖區(qū) buf,并在最后追加一個 null 字符來終止字符串嘿棘。
如果這個函數(shù)在讀取最后一個字符之前就遇到一個換行符 '\n' 或文件的末尾 EOF劲腿,則只會返回讀取到的字符,包括換行符鸟妙。您也可以使用 int fscanf(FILE *fp, const char *format, ...) 函數(shù)來從文件中讀取字符串焦人,但是在遇到第一個空格字符時,它會停止讀取重父。
示例
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <Windows.h>
//讀取文本文件
void main() {
char path[] = "C:\\Users\\Administrator\\Desktop\\friend.txt"; //文本中語句為 hello world
//打開
FILE *fp = fopen(path, "r");
if (fp == NULL)
{
printf("文件打開失敗...");
return;
}
//讀取
char buff[50];//緩沖
while (fgets(buff,50,fp)) {
printf("%s", buff);
}
//關(guān)閉
fclose(fp);
getchar();
}
結(jié)果輸出:
hello world
寫入文件
下面是把字符寫入到流中的最簡單的函數(shù):
int fputc( int c, FILE *fp );
函數(shù) fputc() 把參數(shù) c 的字符值寫入到 fp 所指向的輸出流中花椭。如果寫入成功,它會返回寫入的字符房午,如果發(fā)生錯誤矿辽,則會返回 EOF。您可以使用下面的函數(shù)來把一個以 null 結(jié)尾的字符串寫入到流中:
int fputs( const char *s, FILE *fp );
函數(shù) fputs() 把字符串 s 寫入到 fp 所指向的輸出流中郭厌。如果寫入成功嗦锐,它會返回一個非負(fù)值,如果發(fā)生錯誤沪曙,則會返回 EOF奕污。您也可以使用 int fprintf(FILE *fp,const char *format, ...) 函數(shù)來寫把一個字符串寫入到文件中。嘗試下面的實例:
//寫入文本文件
void main() {
char *path = "C:\\Users\\Administrator\\Desktop\\test.txt";
//打開
FILE *fp = fopen(path, "w");
char *text = "今天天氣不錯\n出去玩吧!";
fputs(text,fp);
//關(guān)閉
fclose(fp);
getchar();
}
在test文本中輸出:
今天天氣不錯
出去玩吧!
注意:請確保您有可用的 /tmp 目錄液走,如果不存在該目錄碳默,則需要在您的計算機上先創(chuàng)建該目錄。
讀寫二進(jìn)制I/O文件
計算機的文件存儲在物理上都是二進(jìn)制缘眶,文本文件和二進(jìn)制之分嘱根,其實是一個人為的邏輯之分。
C讀寫文本文件與二進(jìn)制文件的差別僅僅體現(xiàn)在回車換行符:
1.寫文本時巷懈,每遇到一個'\n'该抒,會將其轉(zhuǎn)換成'\r\n'(回車換行)。
2.讀文本時顶燕,每遇到一個'\r\n'凑保,會將其轉(zhuǎn)換成'\n'冈爹。
3.但是讀寫二進(jìn)制文件的時候并不會做以上轉(zhuǎn)換。
函數(shù)原型:
size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );
其中欧引,ptr:指向保存結(jié)果的指針频伤;size:每個數(shù)據(jù)類型的大小芝此;count:數(shù)據(jù)的個數(shù)憋肖;stream:文件指針
函數(shù)返回讀取數(shù)據(jù)的個數(shù)。
size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );
其中婚苹,ptr:指向保存數(shù)據(jù)的指針岸更;size:每個數(shù)據(jù)類型的大小膊升;count:數(shù)據(jù)的個數(shù)坐慰;stream:文件指針
函數(shù)返回寫入數(shù)據(jù)的個數(shù)。
下面是二進(jìn)制文件讀寫的例子(圖片的復(fù)制):
void main() {
char *read_path = "D:\\BaiduNetdiskDownload\\ndk\\2016_08_08_C_聯(lián)合體_枚舉_IO\\files\\girl.png";
char *write_path = "D:\\BaiduNetdiskDownload\\ndk\\2016_08_08_C_聯(lián)合體_枚舉_IO\\files\\girl_new.png";
//b字符表示操作二進(jìn)制文件binary
FILE *read_fp = fopen(read_path, "rb");
//寫的文件
FILE *write_fp = fopen(write_path,"wb");
//復(fù)制
int buff[50]; //緩沖區(qū)域
int len = 0;//每次讀到的數(shù)據(jù)長度
while ((len = fread(buff,sizeof(int), 50,read_fp))!=0) {//50 是寫的比較大的一個數(shù)
//將讀到的內(nèi)容寫入新的文件
fwrite(buff, sizeof(int), len, write_fp);
}
fclose(read_fp);
fclose(write_fp);
getchar();
}
獲取文件的大小
void main() {
char *read_path = "D:\\BaiduNetdiskDownload\\ndk\\2016_08_08_C_聯(lián)合體_枚舉_IO\\files\\girl.png";
FILE *fp = fopen(read_path, "r");
//重新定位文件指針
//SEEK_END文件末尾用僧,0偏移量
fseek(fp, 0, SEEK_END);
//返回當(dāng)前的文件指針,相對于文件開頭的位移量
long filesize = ftell(fp);
printf("%d\n", filesize);
getchar();
}
文本簡單加密赞咙、解密
void crypt(char normal_path[], char crypt_path[]) {
//打開文件
FILE *normal_fp = fopen(normal_path, "r");
FILE *crypt_fp = fopen(crypt_path, "w");
//一次讀取一個字符
int ch;
while ((ch = fgetc(normal_fp)) != EOF) { //End of File
//寫入(異或運算)
fputc(ch ^ 9, crypt_fp);
}
// 關(guān)閉
fclose(crypt_fp);
fclose(normal_fp);
}
//解密
void decrypt(char crypt_path[],char decrypt_path[]) {
//打開文件
FILE *normal_fp = fopen(crypt_path,"r");
FILE *crypt_fp = fopen(decrypt_path, "w");
//一次讀取一個字符
int ch;
while ((ch = fgetc(normal_fp)) !=EOF)//End of File
{
//寫入(異或運算)
fputc(ch ^ 9, crypt_fp);
}
//關(guān)閉
fclose(crypt_fp);
fclose(normal_fp);
}
void main() {
char *normal_path = "D:\\userinfo.txt";
char *crypt_path = "D:\\userinfo_crypt.txt";
char *decrypt_path = "D:\\userinfo_decrypt.txt";
//加密文件
crypt(normal_path, crypt_path);
//解密文件
decrypt(crypt_path, decrypt_path);
getchar();
}
二進(jìn)制文件簡單加解密
void crypt(char normal_path[], char crypt_path[], char password[]) {
//打開文件
FILE *normal_fp = fopen(normal_path, "rb");
FILE *crypt_fp = fopen(crypt_path, "wb");
//一次讀取一個字符
int ch;
int i = 0; //循環(huán)使用密碼中的字母進(jìn)行異或運算
int pwd_len = strlen(password); //密碼的長度
while ((ch = fgetc(normal_fp)) != EOF) { //End of File
//寫入(異或運算)
fputc(ch ^ password[i % pwd_len], crypt_fp);
i++;
}
//關(guān)閉
fclose(crypt_fp);
fclose(normal_fp);
}
//解密
void decrypt(char crypt_path[], char decrypt_path[], char password[]) {
//打開文件
FILE *normal_fp = fopen(crypt_path, "rb");
FILE *crypt_fp = fopen(decrypt_path, "wb");
//一次讀取一個字符
int ch;
int i = 0; //循環(huán)使用密碼中的字母進(jìn)行異或運算
int pwd_len = strlen(password); //密碼的長度
while ((ch = fgetc(normal_fp)) != EOF) { //End of File
//寫入(異或運算)
fputc(ch ^ password[i % pwd_len], crypt_fp);
i++;
}
//關(guān)閉
fclose(crypt_fp);
fclose(normal_fp);
}
void main() {
char *normal_path = "D:\\girl.png";
char *crypt_path = "D:\\girl_crypt.png";
char *decrypt_path = "D:\\girl_decrypt.png";
//加密文件
crypt(normal_path, crypt_path, "123456");
//加密文件
decrypt(crypt_path, decrypt_path, "123456");
getchar();
}
一般騰訊责循、阿里等大公司的用戶關(guān)鍵數(shù)據(jù)是用C\C++(動態(tài)庫so反編譯很難)加密的。因為Java的加密方法反編譯比較容易破解攀操。
特別感謝:
動腦學(xué)院Jason
菜鳥教程