標準I/O操作

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函數檢驗數據是否寫入文件。


發佈了5 篇原創文章 · 獲贊 12 · 訪問量 6萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章