接續上篇C語言基礎及指針⑦結構體與指針
在結構體與指針中 , 我們了解到結構體與java中的類相似 , 也是一種自定義類型數據結構 , 也學習了結構的各種用法 , 以及簡單的應用 。
在編寫應用程序的時候 , 文件IO操作是不可避免的 , 小到日志本地化收集 , 大到數據格式化存儲 , 都需要使用文件IO , 來操作文件流進行數據處理 。在使用java開發的時候 , 我們有File
類和FileReader
,FileWriter
類來搭配使用 , 也有FileInputStream
,FileOutputStream
和BufferedInputStream
,BufferedOutputStream
金牌組合 。使得java中的文件IO很方便 , 下面我們就來看看簡單的java文件IO示例:
// 讀取文件中的字符
private static void readString() throws Exception{
File _file = new File("e:"+separetor+"a.txt") ;
if (!_file.exists()) {
boolean createStatuts = _file.createNewFile() ;
if (createStatuts) {
System.out.println("創建了一個新文件 ,并且創建成功了");
}else {
System.out.println("創建新文件失敗");
}
}
// 創建輸入流
InputStream fileInputStream = new FileInputStream(_file) ;
byte[] bytes = new byte[1024] ;
int len = -1;
StringBuffer buffer = new StringBuffer() ;
while ((len = fileInputStream.read(bytes)) != -1) {
buffer.append(new String(bytes,0,len)) ;
}
// 關閉輸入流
fileInputStream.close() ;
System.out.println(buffer.toString());
}
// 將字符串寫入文件
private static void writeString() throws Exception{
File _file = new File("e:"+separetor+"a.txt") ;
if (!_file.exists()) {
boolean createStatuts = _file.createNewFile() ;
if (createStatuts) {
System.out.println("創建了一個新文件 ,并且創建成功了");
}else {
System.out.println("創建新文件失敗");
}
}
// 創建輸出流
OutputStream fileOutputStream = new FileOutputStream(_file) ;
String info = "《op 青空 pure rouge 君吻》《君吻 その目に映るもの piano》" ;
fileOutputStream.write(info.getBytes()) ;
// 關閉輸出流
fileOutputStream.close() ;
}
文件IO操作步驟:
1.創建一個File對象(需要操作的文件)
2.構建輸入輸出流
3.創建緩沖區 , 緩存讀寫數據 (將流數據讀入到內存或寫入到磁盤)
3.關閉流 (關閉文件流)
語言都是相通的 , 在C語言中文件IO的操作也是如上述幾步 , 下面我們就一起來看看:
/*讀取文本文件*/
void readTextFile() {
char* path = "C:\\Users\\Zeno\\Documents\\Visual Studio 2015\\Projects\\Hello_C\\Hello_C\\StructPointer.c";
// 打開文件
FILE* fp = fopen(path, "r");
if (fp == NULL)
{
printf("打開文件失敗\n");
return;
}
// 字符緩沖區 , 每次讀一個字符串 , 都會緩存到字符數組中
char buffer[1024];
while (fgets(buffer, 1024, fp)) {
printf("%s", buffer);
}
// 關閉文件流
fclose(fp);
/*寫入文本文件*/
void writeTextFile() {
char* path = "E:\\document\\write.txt";
char* content = "如果 愛情是一場花火 ,一閃即逝的花火,我也要去追求\n如果 愛情是一場花火 ,一閃即逝的花火,我也要去追求\n";
// 打開文件
/*
打開只寫文件,若文件存在則文件長度清為0,即該文件內容會消失。若文件不存在則建立該文件。
*/
FILE* fp = fopen(path, "w");
if (fp == NULL)
{
printf("打開文件失敗\n");
return;
}
// 寫入文件
fputs(content, fp);
// 關閉文件流
fclose(fp);
}
}
首先使用fopen
函數得到一個文件指針 , 操作符r
為讀取文件流 , 構建了一個buffer
數據緩沖區 , 通過fgets
函數循環讀取文件數據 , fclose
函數關閉文件流 。在操作文件IO的時候 , 最重要的函數 , 莫過于fopen
函數了 , 首先我們來看一下fopen
函數的定義:
_ACRTIMP FILE* __cdecl fopen(
_In_z_ char const* _FileName,
_In_z_ char const* _Mode
);
我們發現fopen
函數 , 需要傳入文件全路徑名稱 , 還有一個_Mode
, 這個是文件操作模式 , C語言中文件操作主要依靠操作模式來辨別是輸入流還是輸出流的 。
下面列舉一些常用的操作模式:
mode有下列幾種形態字符串:
r 打開只讀文件,該文件必須存在。
r+ 打開可讀寫的文件,該文件必須存在。
w 打開只寫文件,若文件存在則文件長度清為0,即該文件內容會消失。若文件不存在則建立該文件。
w+ 打開可讀寫文件,若文件存在則文件長度清為零,即該文件內容會消失。若文件不存在則建立該文件。
a 以附加的方式打開只寫文件。若文件不存在,則會建立該文件,如果文件存在,寫入的數據會被加到文件尾,即文件原先的內容會被保留。
a+ 以附加方式打開可讀寫的文件。若文件不存在,則會建立該文件,如果文件存在,寫入的數據會被加到文件尾后,即文件原先的內容會被保留。
值得注意的是 , 上述操作模式是針對文本文件的 , 如果要操作二進制文件 , 則需要在上述操作符后面加上b
, 如rb
,wb
,ab
, 等等 。
不論是文本文件的操作還是字符文件的操作 , 都是 , 打開文件 , 創建緩沖區 , 讀寫文件 。
二進制文件讀寫
/*讀寫二進制文件 -- 復制文件*/
void fileCopy() {
char* currentPath = "E:\\android_pdf\\研磨設計模式.pdf";
char* destPath = "E:\\android_pdf\\研磨設計模式_new.pdf";
// 打開文件
FILE* currentFile_P = fopen(currentPath, "rb");
FILE* destFile_P = fopen(destPath, "wb");
// 先讀取再寫入
int buffer[1024]; // 數據緩沖區
int len; // 每次讀取數據的長度
while ((len = fread(buffer,sizeof(int),1024,currentFile_P)) != EOF)
{
// 將緩沖區里的內容寫入到文件中
fwrite(buffer, sizeof(int), len, destFile_P);
}
// 關閉流
fclose(destFile_P);
fclose(currentFile_P);
}
讀寫二進制和讀寫文本文件沒多少區別 , 最大的區別就是fopen
函數中的模式的不同 , 文本文件是r
,w
, 二進制文件是rb
,wb
。
了解了文件IO的基本操作 , 我們使用文件IO流寫一個加密解密的小程序。我們知道 , 所有的文件都是以二進制存儲的 , 我們看的文本文件, 圖片文件 , 視頻文件 , 都是以二進制存儲在磁盤上的 , 那么 , 我們可以將文件讀取出來 , 進行二進制運算 , 就可以將文件加密解密了 。下面我們通過^
異或運算來進行文件的加密解密 , 異或運算的規則如下:
0 ^ 1 得 1
1 ^ 1 得 0
0 ^ 0 得 0
1 ^ 0 得 1
相同為0 不同為1 , 例如 , 我們將4
這個數加密 , 異或的數是5 , 下面我們來看看運算:
4的二進制是:0100
5的二進制是:0101
異或運算結果(加密):4 ^ 5 == 0001
異或運算結果(解密): 0001 ^ 0101 == 0100
由上述可見 , ^一次為加密 , 再^一次就是解密
代碼示例如下:
文件加密
/*加密文件*/
void encryptFile() {
// 待加密文件路徑
char* normal_path = "E:\\poto\\xj.jpg";
// 加密后文件路徑
char* encrypt_path = "E:\\poto\\xj_encrypt.jpg";
//打開文件
FILE* normal_fp = fopen(normal_path, "rb");
FILE* encrypt_fp = fopen(encrypt_path, "wb");
// 讀文件
int buffer;
while ((buffer = fgetc(normal_fp)) != EOF) {
// 寫入文件
fputc(buffer ^ 8, encrypt_fp);
}
printf("文件加密成功\n");
// 關閉流
fclose(encrypt_fp);
fclose(normal_fp);
}
文件解密
/*文件解密*/
void decryptFile() {
// 加密文件路徑
char* encrypt_path = "E:\\poto\\xj_encrypt.jpg";
// 解密文件路徑
char* decrypt_path = "E:\\poto\\xj_deencrypt.jpg";
// 打開文件
FILE* encrypt_fp = fopen(encrypt_path, "rb");
FILE* decrypt_fp = fopen(decrypt_path, "wb");
// 讀取文件
int buffer;
while ((buffer = fgetc(encrypt_fp)) != EOF) {
// 寫文件
fputc(buffer ^ 8, decrypt_fp);
}
printf("文件解密成功\n");
// 關閉流
fclose(decrypt_fp);
fclose(encrypt_fp);
}
了解了文件加密的原理 , 我們也可以推導出其他形式的加密 , 如帶密碼的文件加密解密 , 混合文件加密解密等等 。
不知不覺C語言基礎系列已經寫了快十篇了 , 也快告一段落了 , 有了這些基礎知識 , 我們就可以去分析分析jni.h頭文件了 , 下一個系列是jni開發系列 , 我們學C語言就是為了能和java打交道 , 那么下個系列我們就來學習C與java的橋梁 , jni (Java Native Interface)技術 。
Android程序員學C系列:
C語言基礎及指針①
C語言基礎及指針②之指針內存分析
C語言基礎及指針③函數與二級指針
C語言基礎及指針④函數指針
C語言基礎及指針⑤動態內存分配
C語言基礎及指針⑥字符操作
C語言基礎及指針⑦結構體與指針
C語言基礎及指針⑧文件IO
C語言基礎及指針⑨聯合體與枚舉
C語言基礎及指針⑩預編譯及jni.h分析