|
例12
#include<stdio.h> main() { char *s, m[20]; int i; FILE *fp; fp=fopen("test.dat", "r"); /*打開文字文件只讀*/ fgets(s, 24, fp); /*從文件中讀取23個字符*/ printf("%s", s); /*輸出所讀的字符串*/ fscanf(fp, "%d", &i); /*讀取整型數*/ printf("%d", i); /*輸出所讀整型數*/ putchar(fgetc(fp)); /*讀取一個字符同時輸出*/ fgets(m, 17, fp); /*讀取16個字符*/ puts(m); /*輸出所讀字符串*/ fclose(fp); /*關閉文件*/ getch(); /*等待任一鍵*/ }
運行后屏幕顯示:
Your score of TOEFL is: 617 That''s good news
如果將上例中fscanf(fp, "%d", &i)改為fscanf(fp, "%s", m), 再將其后的輸出語句改為printf("%s", m), 則可得出同樣的 結果。由此可見Turbo C2. 0 中只要是讀文字文件, 則不論是字符還是數字都將按其ASCII值處理。 另外還要說明的一點就是 fscanf()函數讀到空白符時, 便自動結束,在使用時要特別注意。
3. 文件的隨機讀寫 有時用戶想直接讀取文件中間某處的信息, 若用文件的順序讀寫必須從文件頭開始直到要求的文件位置再讀, 這顯然不方便。 Turbo C2.0提供了一組文件的隨機讀寫函數, 即可以將文件位置指針定位在所要求讀寫的地方直接讀寫。 文件的隨機讀寫函數如下:
int fseek (FILE *stream, long offset, int fromwhere); int fread(void *buf, int size, int count, FILE *stream); int fwrite(void *buf, int size, int count, FILE *stream); long ftell(FILE *stream);
fseek()函數的作用是將文件的位置指針設置到從fromwhere開始的第offset字節的位置上, 其中fromwhere是下列幾個宏定義 之一: 文件位置指針起始計算位置fromwhere ━━━━━━━━━━━━━━━━━━━━━━━━━━━ 符號常數 數值 含義 ─────────────────────────── SEEK_SET 0 從文件開頭 SEEK_CUR 1 從文件指針的現行位置 SEEK_END 2 從文件末尾 ━━━━━━━━━━━━━━━━━━━━━━━━━━━ offset是指文件位置指針從指定開始位置(fromwhere指出的位置)跳過的字節數。它是一個長整型量, 以支持大于64K字節的 文件。fseek()函數一般用于對二進制文件進行操作。 當fseek()函數返回0時表明操作成功, 返回非0表示失敗。 下面程序從二進制文件test_b.dat中讀取第8個字節。
例13:
#include<stdio.h> main() { FILE *fp; if((fp=fopen("test_b.dat", "rb"))==NULL) { printf("Can''t open file"); exit(1); } fseek(fp, 8. 1, SEEK_SET); fgetc(fp); fclose(fp); }
fread()函數是從文件中讀count個字段, 每個字段長度為size個字節, 并把它們存放到buf指針所指的緩沖器中。 fwrite()函數是把buf指針所指的緩沖器中, 長度為size個字節的count個字段寫到stream指向的文件中去。 隨著讀和寫字節數的增大, 文件位置指示器也增大, 讀多少個字節, 文件位置指示器相應也跳過多少個字節。讀寫完畢函數返 回所讀和所寫的字段個數。 ftell()函數返回文件位置指示器的當前值, 這個值是指示器從文件頭開始算起的字節數, 返回的數為長整型數, 當返回-1 時, 表明出現錯誤。 下面程序把一個浮點數組以二進制方式寫入文件test_b.dat中。
例14:
#include <stdio.h> main() { float f[6]={3.2, -4.34, 25.04, 0.1, 50.56, 80.5}; /*定義浮點數組并初始化*/ int i; FILE *fp; fp=fopen("test_b.dat", "wb"); /*創建一個二進制文件只寫*/ fwrite(f, sizeof(float), 6, fp); /*將6個浮點數寫入文件中*/ fclose(fp); /*關閉文件*/ }
下面例子從test_b.dat文件中讀100個整型數, 并把它們放到dat數組中。
例15:
#include <stdio.h> main() { FILE *fp; int dat[100]; fp=fopen("test_b.dat", "rb"); /*打開一個二進制文件只讀*/ if(fread(dat, sizeof(int), 100, fp)!=100) /*判斷是否讀了100個數*/ { if(feof(fp)) printf("End of file"); /*不到100個數文件結束*/ else printf("Read error"); /*讀數錯誤*/ fclose(fp); /*關閉文件*/ }
注意: 當用標準文件函數對文件進行讀寫操作時, 首先將所讀寫的內容放進緩沖區, 即寫函數只對輸出緩沖區進行操作, 讀函數只對 輸入緩沖區進行操作。例如向一個文件寫入內容, 所寫的內容將首先放在輸出緩沖區中, 直到輸出緩沖區存滿或使用fclose()函數 關閉文件時, 緩沖區的內容才會寫入文件中。 若無fclose() 函數, 則不會向文件中存入所寫的內容或寫入的文件內容不全。有一 個對緩沖區進行刷新的函數, 即fflush(), 其調用格式為:
int fflush(FILE *stream);
該函數將輸出緩沖區的內容實際寫入文件中, 而將輸入緩沖區的內容清除掉。
4. feof()和rewind()函數 這兩個函數的調用格式為:
int feof(FILE *stream); int rewind(FILE *stream);
feof()函數檢測文件位置指示器是否到達了文件結尾, 若是則返回一個非0值, 否則返回0。這個函數對二進制文件操作特別 有用, 因為二進制文件中, 文件結尾標志EOF也是一個合法的二進制數, 只簡單的檢查讀入字符的值來判斷文件是否結束是不行的。 如果那樣的話, 可能會造成文件未結尾而被認為結尾, 所以就必須有feof()函數。 下面的這條語句是常用的判斷文件是否結束的方法。
while(!feof(fp)) fgetc(fp);
while為循環語句, 將在下面介紹。 rewind()函數用于把文件位置指示器移到文件的起點處, 成功時返回0, 否則, 返回非0值。
1.2.2 非標準文件函數 這類函數最早用于UNIX操作系統, ANSI標準未定義, 但有時也經常用到, DOS 3.0以上版本支持這些函數。它們的頭文件為 io.h。
一、文件的打開和關閉
1. open()函數 open()函數的作用是打開文件, 其調用格式為:
int open(char *filename, int access);
該函數表示按access的要求打開名為filename的文件, 返回值為文件描述字, 其中access有兩部分內容: 基本模式和修飾符, 兩者用" "("或")方式連接。修飾符可以有多個, 但基本模式只能有一個。access的規定如表3-2。
表3-2 access的規定 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 基本模式 含義 修飾符 含 義 ──────────────────────────── O_RDONLY 只讀 O_APPEND 文件指針指向末尾 O_WRONLY 只寫 O_CREAT 文件不存在時創建文件, 屬性按基本模式屬性 O_RDWR 讀寫 O_TRUNC 若文件存在, 將其長度 縮為0, 屬性不變 O_BINARY 打開一個二進制文件 O_TEXT 打開一個文字文件 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━ open()函數打開成功, 返回值就是文件描述字的值(非負值), 否則返回-1。
2. close()函數 close()函數的作用是關閉由open()函數打開的文件, 其調用格式為:
int close(int handle);
該函數關閉文件描述字handle相連的文件。
二、讀寫函數
1. read()函數 read()函數的調用格式為:
int read(int handle, void *buf, int count);
read()函數從handle(文件描述字)相連的文件中, 讀取count個字節放到buf所指的緩沖區中, 返回值為實際所讀字節數, 返回 -1表示出錯。返回0 表示文件結束。
2. write()函數 write()函數的調用格式為: int write(int handle, void *buf, int count); write()函數把count個字節從buf指向的緩沖區寫入與handle相連的文件中, 返回值為實際寫入的字節數。
三、隨機定位函數
1. lseek()函數 lseek()函數的調用格式為:
int lseek(int handle, long offset, int fromwhere);
該函數對與handle相連的文件位置指針進行定位, 功能和用法與fseek() 函數相同。
2. tell()函數 tell()函數的調用格式為:
long tell(int handle);
該函數返回與handle相連的文件現生位置指針, 功能和用法與ftell()相同。
|