FILE *fopen(const char *filename, const char *mode);
fopen函數由filename打開,mode參數指定文件的打開方式:
'r' 只讀方式打開,將文件指針指向文件頭,如果文件不存在,則File返回空。
'r+' 讀寫方式打開,將文件指針指向文件頭,如果文件不存在,則File返回空。'w' 寫入方式打開,將文件指針指向文件頭並將文件大小截爲零。如果文件不存在則嘗試創建之。
'w+' 讀寫方式打開,將文件指針指向文件頭並將文件大小截爲零。如果文件不存在則嘗試創建之。
'a' 寫入方式打開,將文件指針指向文件末尾。如果文件不存在則嘗試創建之。
'a+' 讀寫方式打開,將文件指針指向文件末尾。如果文件不存在則嘗試創建之。
'x' 創建並以寫入方式打開,將文件指針指向文件頭。如果文件已存在,則 fopen() 調用失敗並返回 FALSE。
'x' 創建並以寫入方式打開,將文件指針指向文件頭。如果文件已存在,則 fopen() 調用失敗並返回 FALSE。
'b' 使用字符b作爲文件類型的判斷,是否是binary文件。
fseek函數用於在文件流裏爲下一次讀寫操作指定位置。
通常用的幾個如下:
讀寫位置移動到文件開頭:fseek(FILE *file,0,SEEK_SET);
讀寫位置移動到文件末尾:fseek(FILE *file,0,SEEK_END);
取得當前文件位置:fseek(FILE *file,0,SEEK_CUR);
fwrite和fread是以記錄爲單位的I/O函數,fread和fwrite函數一般用於二進制文件的輸入輸出。
#include <stdio.h>
size_t fread(void *ptr, size_t size, size_t number, FILE *stream);
size_t fwrite(const void *ptr, size_t size, size_t number, FILE *stream);
返回值:讀或寫的記錄數,成功時返回的記錄數等於number,出錯或讀到文件末尾時返回的記錄數小於number,也可能返回0。
fread和fwrite用於讀寫記錄,這裏的記錄是指一串固定長度的字節,比如一個int、一個結構體或者一個定長數組。參數size指出一條記錄的長度,而number指出要讀或寫多少條記錄,這些記錄在ptr所指的內存空間中連續存放,共佔size * number個字節,fread從文件stream中讀出size * number個字節保存到ptr中,而fwrite把ptr中的size * number個字節寫到文件stream中。number是請求讀或寫的記錄數,fread和fwrite返回的記錄數有可能小於nmemb指定的記錄數。例如當前讀寫位置距文件末尾只有一條記錄的長度,調用fread時指定number爲2,則返回值爲1。如果當前讀寫位置已經在文件末尾了,或者讀文件時出錯了,則fread返回0。如果寫文件時出錯了,則fwrite的返回值小於number指定的值。下面的例子由兩個程序組成,一個程序把結構體保存到文件中,另一個程序和從文件中讀出結構體
fread和fwrite的例子程序如下:
/* --writefile.c-- */
#include <stdio.h>
#include <stdlib.h>
struct record {
char name[10];
int age;
};
int main(void)
{
struct record array[2] = {{"Ken", 24}, {"Knuth", 28}};
FILE *fp = fopen("file", "w");
if (fp == NULL) {
perror("Open file fail");
exit(1);
}
fwrite(array, sizeof(struct record), 2, fp);
fclose(fp);
return 0;
}
/* --readfile.c-- */
#include <stdio.h>
#include <stdlib.h>
struct record {
char name[10];
int age;
};
int main(void)
{
struct record array[2];
FILE *fp = fopen("file", "r");
if (fp == NULL) {
perror("Open file fail");
exit(1);
}
fread(array, sizeof(struct record), 2, fp);
printf("Name1: %s\tAge1: %d\n", array[0].name, array[0].age);
printf("Name2: %s\tAge2: %d\n", array[1].name, array[1].age);
fclose(fp);
return 0;
}
編譯後發現生成的文件file不能直接打開。原因:我們把一個struct record結構體看作一條記錄,由於結構體中有填充字節,每條記錄佔16字節,
把兩條記錄寫到文件中共佔32字節。該程序生成的file文件是二進制文件而非文本文件,因爲其
中不僅保存着字符型數據,還保存着整型數據24和28(在od命令的輸出中以八進制顯示爲030和034)。
注意,直接在文件中讀寫結構體的程序是不可移植的,如果在一種平臺上編譯運行writebin.c程序,
把生成的recfile文件拷到另一種平臺並在該平臺上編譯運行readbin.c程序,則不能保證正確讀出
文件的內容,因爲不同平臺的大小端可能不同(因而對整型數據的存儲方式不同),結構體的填充方式
也可能不同(因而同一個結構體所佔的字節數可能不同,age成員在name成員之後的什麼位置也可能不同)。
通過readfile程序讀取文件file的內容,說明writefile程序的確記錄成功寫入file中。
從file讀出的內容如下:
Name1: Ken Age1: 24
Name2: Knuth Age2: 28
fwrite和fread的應用舉例:
1.將一個字符串寫入文件:
char *str="hello,Iam a test program!";
fwrite(str,sizeof(char),strlen(str),fp);
2.將一個字符數組寫入文件:
char str[]={'a','b,'c'};
fwrite(str,sizeof(char),sizeof(str),fp);
3.將一個整型數組寫入文件:
int a[]={12,33,23,24,12};
先計算數組元素個數number,之後
fwrite(a,sizeof(int),number,fp)
注:由於程序生成的文件是二進制文件而非文本文件,因此,不用機器,整數的表達不同,所以無法直接打開生成文件。可通過fread函數檢驗數據是否寫入文件。