Android NDK開(kāi)發(fā)(C語(yǔ)言-文件讀寫(xiě))
1.文件讀寫(xiě)
一個(gè)文件,無(wú)論它是文本文件還是二進(jìn)制文件,都是代表了一系列的字節(jié)。C 語(yǔ)言不僅提供了訪問(wèn)頂層的函數(shù),也提供了底層(OS)調(diào)用來(lái)處理存儲(chǔ)設(shè)備上的文件。
1.1打開(kāi)文件
我們可以使用 fopen( ) 函數(shù)來(lái)創(chuàng)建一個(gè)新的文件或者打開(kāi)一個(gè)已有的文件,這個(gè)調(diào)用會(huì)初始化類(lèi)型 FILE 的一個(gè)對(duì)象,類(lèi)型 FILE 包含了所有用來(lái)控制流的必要的信息。
下面是這個(gè)函數(shù)調(diào)用的原型:
FILE *fopen( const char * filename, const char * mode );
在這里,filename 是字符串,用來(lái)命名文件,訪問(wèn)模式 mode 的值可以是下列值中的一個(gè):
| 模式 | 描述 |
|---|---|
| r | 打開(kāi)一個(gè)已有的文本文件,允許讀取文件。 |
| w | 打開(kāi)一個(gè)文本文件,允許寫(xiě)入文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。在這里,您的程序會(huì)從文件的開(kāi)頭寫(xiě)入內(nèi)容。 |
| a | 打開(kāi)一個(gè)文本文件,以追加模式寫(xiě)入文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。在這里,您的程序會(huì)在已有的文件內(nèi)容中追加內(nèi)容。 |
| r+ | 打開(kāi)一個(gè)文本文件,允許讀寫(xiě)文件。 |
| w+ | 打開(kāi)一個(gè)文本文件,允許讀寫(xiě)文件。如果文件已存在,則文件會(huì)被截?cái)酁榱汩L(zhǎng)度,如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。 |
| a+ | 打開(kāi)一個(gè)文本文件,允許讀寫(xiě)文件。如果文件不存在,則會(huì)創(chuàng)建一個(gè)新文件。讀取會(huì)從文件的開(kāi)頭開(kāi)始,寫(xiě)入則只能是追加模式。 |
如果處理的是二進(jìn)制文件,則需使用下面的訪問(wèn)模式來(lái)取代上面的訪問(wèn)模式:
"rb", "wb", "ab", "rb+", "r+b", "wb+", "w+b", "ab+", "a+b"
1.2關(guān)閉文件
為了關(guān)閉文件,請(qǐng)使用 fclose( ) 函數(shù)。
函數(shù)的原型如下:
int fclose( FILE *fp );
如果成功關(guān)閉文件,fclose( ) 函數(shù)返回零,如果關(guān)閉文件時(shí)發(fā)生錯(cuò)誤,函數(shù)返回 EOF。這個(gè)函數(shù)實(shí)際上,會(huì)清空緩沖區(qū)中的數(shù)據(jù),關(guān)閉文件,并釋放用于該文件的所有內(nèi)存。EOF 是一個(gè)定義在頭文件 stdio.h 中的常量。
C 標(biāo)準(zhǔn)庫(kù)提供了各種函數(shù)來(lái)按字符或者以固定長(zhǎng)度字符串的形式讀寫(xiě)文件。
1.3讀取文件
下面是從文件讀取單個(gè)字符的最簡(jiǎn)單的函數(shù):
int fgetc( FILE * fp );
fgetc() 函數(shù)從 fp 所指向的輸入文件中讀取一個(gè)字符。返回值是讀取的字符,如果發(fā)生錯(cuò)誤則返回 EOF。下面的函數(shù)允許您從流中讀取一個(gè)字符串:
char *fgets( char *buf, int n, FILE *fp );
函數(shù) fgets() 從 fp 所指向的輸入流中讀取 n - 1 個(gè)字符。它會(huì)把讀取的字符串復(fù)制到緩沖區(qū) buf,并在最后追加一個(gè) null 字符來(lái)終止字符串。
如果這個(gè)函數(shù)在讀取最后一個(gè)字符之前就遇到一個(gè)換行符 '\n' 或文件的末尾 EOF,則只會(huì)返回讀取到的字符,包括換行符。您也可以使用 int fscanf(FILE *fp, const char *format, ...) 函數(shù)來(lái)從文件中讀取字符串,但是在遇到第一個(gè)空格字符時(shí),它會(huì)停止讀取。
示例:
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <Windows.h>
//讀取文本文件
void main() {
char path[] = "C:\Users\Administrator\Desktop\friend.txt"; //文本中語(yǔ)句為 hello world
//打開(kāi)
FILE *fp = fopen(path, "r");
if (fp == NULL)
{
printf("文件打開(kāi)失敗...");
return;
}
//讀取
char buff[50];//緩沖
while (fgets(buff,50,fp)) {
printf("%s", buff);
}
//關(guān)閉
fclose(fp);
getchar();
}
結(jié)果輸出:
?hello world
1.4寫(xiě)入文件
下面是把字符寫(xiě)入到流中的最簡(jiǎn)單的函數(shù):
int fputc( int c, FILE *fp );
函數(shù) fputc() 把參數(shù) c 的字符值寫(xiě)入到 fp 所指向的輸出流中。如果寫(xiě)入成功,它會(huì)返回寫(xiě)入的字符,如果發(fā)生錯(cuò)誤,則會(huì)返回 EOF。
我們可以使用下面的函數(shù)來(lái)把一個(gè)以 null 結(jié)尾的字符串寫(xiě)入到流中:
int fputs( const char *s, FILE *fp );
函數(shù) fputs() 把字符串 s 寫(xiě)入到 fp 所指向的輸出流中。如果寫(xiě)入成功,它會(huì)返回一個(gè)非負(fù)值,如果發(fā)生錯(cuò)誤,則會(huì)返回 EOF。您也可以使用 int fprintf(FILE *fp,const char *format, ...) 函數(shù)來(lái)寫(xiě)把一個(gè)字符串寫(xiě)入到文件中。
嘗試下面的實(shí)例:
//寫(xiě)入文本文件
void main() {
char *path = "C:\Users\Administrator\Desktop\test.txt";
//打開(kāi)
FILE *fp = fopen(path, "w");
char *text = "今天天氣不錯(cuò)\n出去玩吧!";
fputs(text,fp);
//關(guān)閉
fclose(fp);
getchar();
}
在test文本中輸出:
今天天氣不錯(cuò)
出去玩吧!
注意:請(qǐng)確保您有可用的 /tmp 目錄,如果不存在該目錄,則需要在您的計(jì)算機(jī)上先創(chuàng)建該目錄。
1.5讀寫(xiě)二進(jìn)制I/O文件
計(jì)算機(jī)的文件存儲(chǔ)在物理上都是二進(jìn)制,文本文件和二進(jìn)制之分,其實(shí)是一個(gè)人為的邏輯之分。
C讀寫(xiě)文本文件與二進(jìn)制文件的差別僅僅體現(xiàn)在回車(chē)換行符:
- 1.寫(xiě)文本時(shí),每遇到一個(gè)'\n',會(huì)將其轉(zhuǎn)換成'\r\n'(回車(chē)換行)。
- 2.讀文本時(shí),每遇到一個(gè)'\r\n',會(huì)將其轉(zhuǎn)換成'\n'。
- 3.但是讀寫(xiě)二進(jìn)制文件的時(shí)候并不會(huì)做以上轉(zhuǎn)換。
函數(shù)原型:
size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );
其中:
ptr:指向保存結(jié)果的指針;size:每個(gè)數(shù)據(jù)類(lèi)型的大?。?/li>count:數(shù)據(jù)的個(gè)數(shù);stream:文件指針
函數(shù)返回讀取數(shù)據(jù)的個(gè)數(shù)。
size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );
其中,ptr:指向保存數(shù)據(jù)的指針;size:每個(gè)數(shù)據(jù)類(lèi)型的大小;count:數(shù)據(jù)的個(gè)數(shù);stream:文件指針
函數(shù)返回寫(xiě)入數(shù)據(jù)的個(gè)數(shù)。
下面是二進(jìn)制文件讀寫(xiě)的例子(圖片的復(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");
//寫(xiě)的文件
FILE *write_fp = fopen(write_path,"wb");
//復(fù)制
int buff[50]; //緩沖區(qū)域
int len = 0;//每次讀到的數(shù)據(jù)長(zhǎng)度
while ((len = fread(buff,sizeof(int), 50,read_fp))!=0) {//50 是寫(xiě)的比較大的一個(gè)數(shù)
//將讀到的內(nèi)容寫(xiě)入新的文件
fwrite(buff, sizeof(int), len, write_fp);
}
fclose(read_fp);
fclose(write_fp);
getchar();
}
1.6獲取文件的大小
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)前的文件指針,相對(duì)于文件開(kāi)頭的位移量
long filesize = ftell(fp);
printf("%d\n", filesize);
getchar();
}
1.7文本簡(jiǎn)單加密、解密
void crypt(char normal_path[], char crypt_path[]) {
//打開(kāi)文件
FILE *normal_fp = fopen(normal_path, "r");
FILE *crypt_fp = fopen(crypt_path, "w");
//一次讀取一個(gè)字符
int ch;
while ((ch = fgetc(normal_fp)) != EOF) { //End of File
//寫(xiě)入(異或運(yùn)算)
fputc(ch ^ 9, crypt_fp);
}
// 關(guān)閉
fclose(crypt_fp);
fclose(normal_fp);
}
//解密
void decrypt(char crypt_path[],char decrypt_path[]) {
//打開(kāi)文件
FILE *normal_fp = fopen(crypt_path,"r");
FILE *crypt_fp = fopen(decrypt_path, "w");
//一次讀取一個(gè)字符
int ch;
while ((ch = fgetc(normal_fp)) !=EOF)//End of File
{
//寫(xiě)入(異或運(yùn)算)
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();
}
1.8二進(jìn)制文件簡(jiǎn)單加解密
void crypt(char normal_path[], char crypt_path[], char password[]) {
//打開(kāi)文件
FILE *normal_fp = fopen(normal_path, "rb");
FILE *crypt_fp = fopen(crypt_path, "wb");
//一次讀取一個(gè)字符
int ch;
int i = 0; //循環(huán)使用密碼中的字母進(jìn)行異或運(yùn)算
int pwd_len = strlen(password); //密碼的長(zhǎng)度
while ((ch = fgetc(normal_fp)) != EOF) { //End of File
//寫(xiě)入(異或運(yùn)算)
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[]) {
//打開(kāi)文件
FILE *normal_fp = fopen(crypt_path, "rb");
FILE *crypt_fp = fopen(decrypt_path, "wb");
//一次讀取一個(gè)字符
int ch;
int i = 0; //循環(huán)使用密碼中的字母進(jìn)行異或運(yùn)算
int pwd_len = strlen(password); //密碼的長(zhǎng)度
while ((ch = fgetc(normal_fp)) != EOF) { //End of File
//寫(xiě)入(異或運(yùn)算)
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++(動(dòng)態(tài)庫(kù)so反編譯很難)加密的。因?yàn)镴ava的加密方法反編譯比較容易破解。
到此這篇關(guān)于Android NDK開(kāi)發(fā)(C語(yǔ)言-文件讀寫(xiě))的文章就介紹到這了,更多相關(guān)C語(yǔ)言-文件讀寫(xiě)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解Android中的沉浸式狀態(tài)欄效果實(shí)例
本篇文章主要介紹了Android中的沉浸式狀態(tài)欄效果,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12
Compose自定義View實(shí)現(xiàn)繪制Rainbow運(yùn)動(dòng)三環(huán)效果
這篇文章主要為大家介紹了一個(gè)基于Compose自定義的一個(gè)Rainbow彩虹運(yùn)動(dòng)三環(huán),業(yè)務(wù)上類(lèi)似于iWatch上的那個(gè)運(yùn)動(dòng)三環(huán),感興趣的小伙伴可以了解一下2023-02-02
Android開(kāi)發(fā)基礎(chǔ)使用ProgressBar加載進(jìn)度條示例
這篇文章主要介紹了安卓開(kāi)發(fā)基礎(chǔ)使用ProgressBar加載進(jìn)度條示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02
Android 使用Picasso加載網(wǎng)絡(luò)圖片等比例縮放的實(shí)現(xiàn)方法
在做android圖片加載的時(shí)候,由于手機(jī)屏幕受限,很多大圖加載過(guò)來(lái)的時(shí)候,我們要求等比例縮放,接下來(lái)小編給大家?guī)?lái)了Android 使用Picasso加載網(wǎng)絡(luò)圖片等比例縮放的實(shí)現(xiàn)方法,感興趣的朋友一起看看吧2018-08-08
Android實(shí)現(xiàn)內(nèi)存中數(shù)據(jù)保存到sdcard的方法
這篇文章主要介紹了Android實(shí)現(xiàn)內(nèi)存中數(shù)據(jù)保存到sdcard的方法,涉及Android的文件讀寫(xiě)與I/O操作相關(guān)技巧,需要的朋友可以參考下2016-01-01
Android編程之監(jiān)聽(tīng)器的實(shí)現(xiàn)方法
這篇文章主要介紹了Android編程之監(jiān)聽(tīng)器的實(shí)現(xiàn)方法,以實(shí)例形式較為詳細(xì)的分析了Android監(jiān)聽(tīng)器的創(chuàng)建、注冊(cè)及相關(guān)使用技巧,需要的朋友可以參考下2015-11-11
Android 詳解自定義圓角輸入框和按鈕的實(shí)現(xiàn)流程
對(duì)于安卓程序員來(lái)說(shuō),自定義view簡(jiǎn)直不要太重要,畢竟有很多功能,譬如圓形頭像這些,用單純的原生非常難以實(shí)現(xiàn),而用自定義view,簡(jiǎn)直分分鐘,今天我們來(lái)實(shí)現(xiàn)自定義圓角輸入框和按鈕,大家可以跟著練習(xí),掌握技巧2021-11-11
Android Activity與Service通信(不同進(jìn)程之間)詳解
這篇文章主要介紹了Android Activity與Service通信(不同進(jìn)程之間)的相關(guān)資料,這里提供了三種方法,需要的朋友可以參考下2016-10-10

