讀寫磁盤文件專題
采用C/C++/MFC/WIN32(API)方式
=================================================================
在Windows平臺上,讀寫磁盤文件是相當多應用程序經常會涉及到的一種功能。該主題涉及到采用C/C++/MFC/Win32(API)中提供的接口函數來操作磁盤文件的方法,以及其中需要注意的地方。
=============================================================
0. 磁盤文件數據存儲方式
在介紹各種操作文件方式之前,需要先介紹磁盤上文件數據的組織方式。
實際上,文件是在計算機內存中以二進制表示的數據在外部存儲介質上的另一種存放形式。文件通常分為二進制文件和文本文件。二進制文件是包含在ASCII及擴展ASCII字符中編寫的數據或程序指令的文件。一般是可執行文件(Exe)、圖形、圖像、聲音等文件。而文本文件(通常也成為ASCII文件),它的每一字節存放的是可表示為一個字符的ASCII代碼的文件。這里把文件區分為二進制文件和文本文件,但實際上它們都是以二進制數據的方式來存儲的。文本文件里所存儲的每一個字節都可以轉化為一個可讀的字符。譬如’a’,’b’,但在內存中并不會存儲'a', 'b',而是存儲它們的ASCII碼:61和62。
1. C語言對文件操作的支持
在C語言中,對于文件的操作都是利用FILE結構體來進行的。包括文件的打開、文件的讀寫、文件的關閉、文件指針的定位等。
(1) 文件的打開
文件的打開需要用fopen函數。該函數帶2個參數,返回值為指向之前定義的FILE結構體指針。語法定義為
FILE* fopen(const char* filename, const char* mod);
參數1:表示要打開文件的完整路徑,例如: “C:\\Test\\Zero_Test.txt”.
參數2:打開文件的方式。取值為下表所示。
文件打開模式
|
意義
|
r/rb
|
為讀取而打開。如果文件不存在或不能找到,函數調用失敗。
|
w/wb
|
為寫入而打開一個空文件。如果給定的文件已經存在,則清空其內容。
|
a/ab
|
為寫入而打開一個文件。如果文件存在,則在文件尾部添加新數據,在寫新數據之前不會移除原有的EOF標志。如果文件不存在,則新建一個空文件以待寫入。
|
r+/rb+
|
打開文件用于寫入操作和讀取操作,文件必須存在。
|
w+/wb+
|
為寫入操作和讀取操作打開一個空文件。如果文件已經存在,則清空其內容。
|
a+/ab+
|
打開文件用于讀取操作和添加操作。并且添加操作在添加新數據之前會移除該文件中已有的EOF標志,然后當寫入操作完成之后再恢復EOF標志。如果指定文件不存在,那么首先將新建一個文件。
|
在上表中,沒有帶b的模式表示打開的是文本文件,而帶b的模式表示打開的是二進制文件。
通常在定義結構FILE時會將其初始化為NULL。在打開文件將函數fopen的返回值賦值給它。之后需要判斷文件是否成功打開,可采用如下方式:
if ((pFile = fopen(“C:\\Test\\Zero_Test.txt”, “w”) == NULL)
Print(“Opening File Error.”); // FILE* pFile = NULL;
else
正常寫入數據到文件…
(2)文件的關閉
在使用完一個文件之后應該關閉它,以防止它再被誤用。“關閉”就是使文件指針變量不再指向該文件,使其脫鉤。函數使用方式:
fclose(pFile);
需要注意的是fclose的參數必須是有效的文件指針變量,否則運行時會掛掉。
應該養成在文件數據操作完成后關閉文件的習慣,如果不關閉文件將會丟失數據。同時,在向文件寫數據時,是現將數據輸出到緩沖區,待緩沖區充滿后才正式輸出給文件。如果程序運行結束而緩沖區并未充滿,則緩沖區中的數據將會丟失。因此利用函數fclose來關閉文件可以避免這個問題。
當然,還有另外一個函數fflush,也可以用來將緩沖區里的數據輸出到文件中。函數調用方式:
fflush(pFile);
(3)文件的讀寫
當涉及到大量數據的讀寫時,可以采用函數fread和fwrite。這兩個函數通常用于二進制文件的讀寫。函數調用方式如下:
fread(buffer, size, count, fp);
fwrite(buffer, size, count, fp);
buffer是一個指針,是用來存放數據的變量指針,例如內置類型變量的指針,結構體變量指針等。
size是要讀寫的字節數。特別要注意的是,此處并不是數組的大小,結構體內變量的個數等。最好采用sizeof操作符來求得buffer的內存字節數。
count是size大小的重復次數。
fp是文件指針。
對于這樣的結構體:
typedef struct {
int clr_sel; // color mode的選擇
int fixval_sel; // fixed value的選擇
float thres_val; // threshold value的確定
float thres_scrlbar_pos; // threshold scroll bar位置的確定
int method_sel; // analysis method的選擇
int usediff_sel; // use difference of MSA/TSE方式的選擇
int usescat_sel; // use MSA/TSE of scatter signal 方式的選擇
int remxtalk_sel; // remove crosstalk 方式的選擇
float timewin_scrlbar_pos; // time window scroll bar位置的確定
int timewin_sel; // time window模式的選擇
int freq_sel; // frequency selection模式的選擇
int path_sel; // path selection模式的選擇
} IMG_SETTINGS;
IMG_SETTINGS m_img_settings;
文件讀寫操作時,可采用如下方式:
fwrite(&m_img_settings, sizeof(m_img_settings), 1, pwFile);
fread(&m_img_settings, sizeof(m_img_settings), 1, prFile);
====================================================
如果想讀寫特定格式的文件,可采用fprintf和fscanf函數。這兩個函數通常是針對文本文件進行操作。
函數調用方式如下:
fprintf(pFile, “%d…”, i, j, k …);
fscanf(pFile, “%d…”, i j, k…);
格式化時可以規定讀寫數據的精度,類型,以及數據之間的分割符等。
例如:
fprintf(pFile, " %d %d %d %d %d %d %d %d \n", sigSize, samp_points, samp_rate,40, avrag_num, 0, 0, 0);
fprintf(pFile, "%d %.2f %s %d %d %d %d \n", vpathdef[i].frequency1/1000,
vpathdef[i].amplitude, sig_type.c_str(), psnset->sensorArray[vpathdef[i].actuator-1].channel-tol,
psnset->sensorArray[vpathdef[i].sensor-1].channel-tol, vpathdef[i].gain, 30);
特別需要注意的是,如果寫入的數據的類型相同時,不可為了簡便,將格式化的字符串寫成如此形式(.., ”%d”,i,j,k…); // i,j,k…同整型類型。
如果這樣操作的話,讀寫的數據便只有數據i了。
最后針對一個面試題來對C語言操作文件的方式作一個總結.
面試題:給你一個整數,例如12345,將這個整數保存到文件中,要求在以記事本打開該文件時,顯示的是:12345。
給出三種代碼:
(1) 代碼1
FILE* pwFile = NULL;
pwFile = fopen(“c:\\Test.txt”, “w”); // create and open file with text mode.
int i = 12345; // However, write number with binary mode.
fwrite(&i, 4, 1, pwFile); // sizeof(int) = 4.
fclose(fwFile);
========================
(2) 代碼2
FILE* pwFile = NULL;
pwFile = fopen(“c:\\Test.txt”, “w”);
char ch[5] = {1+48, 2+48, 3+48, 4+48, 5+48 };
fwrite(ch, 1, 5, pwFile); // sizeof(char) = 1, 5*sizeof(char) = 5
fclose(pwFile);
=======================
(3) 代碼3
FILE* pwFile = NULL;
pwFile = fopen(“c:\\Test.txt”, “w”);
int i = 12345;
char ch[5];
itoa(i, ch, 10); // transform int number to string
fwrite(ch, 1, 5, fwFile);
fclose(pFile);
==============================
(4) 代碼4
FILE* pwFile = NULL;
pwFile = fopen(“c:\\Test.txt”, “w”);
int i = 12345;
fprintf(pwFile, “%d”, i); // write number with specified format.
fclose(pwFile);
==============================
將4種代碼在vc中進行編譯運行,可以發現只有方式(1)不滿足題目要求。
to be continued...