鍵盤、顯示器、打印機、磁盤驅動器等邏輯設備, 其輸入輸出都可以通過文
件管理的方法來完成。
Turbo C2.0提供了兩類關於文件的函數。
- 一類稱做標準文件函數也稱 緩衝型文件函數, 這是ANSI標準定義的函數;
- 另一類叫非標準文件函數, 也稱非緩衝型文件函數。
標準文件函數
標準文件函數主要包括文件的打開、關閉、讀和寫等函數。不象BASIC 、
FORTRAN語方有順序文件和隨機文件之分, 在打開時就應按不同的方式確定。
Turbo C2.0並不區分這兩種文件, 但提供了兩組函數, 即順序讀寫函數和隨機讀
寫函數。
一、文件的打開和關閉
任何一個文件在使用之前和使用之後, 必須要進行打開和關閉, 這是因爲操
作系統對於同時打開的文件數目是有限制的,
DOS操作系統中, 可以在DEVICE .SYS中定義允許同時打開的文件數n(用files=n定義)。其中n 爲可同時打開的文
件數, 一般n<=20。因此在使用文件前應打開文件, 纔可對其中的信息進行存取。用完之後需要關閉, 否則將會出現一些意想不到的錯誤。
Turbo C2.0提供了打開和關閉文件的函數。
1. fopen()函數
fopen函數用於打開文件, 其調用格式爲:
FILE *fopen(char *filename, *type);
在介紹這個函數之;前, 先了解一下下面的知識。
(1) 流(stream)和文件(file)
流和文件在Turbo C2.0中是有區別的, Turbo C2.0 爲編程者和被訪問的設
備之間提供了一層抽象的東西, 稱之爲"流", 而將具體的實際設備叫做文件。
流是一個邏輯設備, 具有相同的行爲。因此, 用來進行磁盤文件寫的函數也同樣
可以用來進行打印機的寫入。
在Turbo C2.0中有兩種性質的流: 文字流( text stream)和二進制(binary stream)。
對磁盤來說就是文本文件和二進制文件。
(2) 文件指針FILE
實際上FILE是一個新的數據類型。它是Turbo C2.0的基本數據類型的集合,
稱之爲結構指針, 即在打開文件時必須先定義一個文件指針。
現在再來看打開文件函數的用法。
fopen()函數中第一個形式參數表示文件名, 可以包含路徑和文件名兩部分。
如:
"B:TEST.DAT"
"C:\\TC\\TEST.DAT"
如果將路徑寫成"C:\TC\TEST.DAT"是不正確的, 這一點要特別注意。
第二個形式參數表示打開文件的類型。關於文件類型的規定參見下表。
表 文件操作類型
━━━━━━━━━━━━━━━━━━━━━━━━━━━━
字符 含義
────────────────────────────
"r" 打開文字文件只讀
"w" 創建文字文件只寫
"a" 增補, 如果文件不存在則創建一個
"r+" 打開一個文字文件讀/寫
"w+" 創建一個文字文件讀/寫
"a+" 打開或創建一個文件增補
"b" 二進制文件(可以和上面每一項合用)
"t" 文這文件(默認項)
━━━━━━━━━━━━━━━━━━━━━━━━━━━━
如果要打開一個CCDOS子目錄中, 文件名爲CLIB的二進制文件, 可寫成:
fopen("c:\\ccdos\\clib", "rb");
如果成功的打開一個文件, fopen()函數返回文件指針, 否則返回空指針
(NULL)。由此可判斷文件打開是否成功。
2. fclose()函數
fclose()函數用來關閉一個由fopen()函數打開的文件 , 其調用格式爲:
int fclose(FILE *stream);
該函數返回一個整型數。當文件關閉成功時, 返回0, 否則返回一個非零值。
可以根據函數的返回值判斷文件是否關閉成功。
例10:
#iclude<stdio.h>
main()
{
FILE *fp; /*定義一個文件指針*/
int i;
fp=fopen("CLIB", "rb"); /*打開當前目錄名爲CLIB的文件只讀*/
if(fp==NULL) /*判斷文件是否打開成功*/
puts("File open error");/*提示打開不成功*/
i=fclose(fp); /*關閉打開的文件*/
if(i==0) /*判斷文件是否關閉成功*/
printf("O,K"); /*提示關閉成功*/
else
puts("File close error");/*提示關閉不成功*/
}
二、有關文件操作的函數
本節所講的文件讀寫函數均是指順序讀寫, 即讀寫了一條信息後, 指針自動
加1。下面分別介紹寫操作函數和讀操作函數。
1. 文件的順序寫函數
fprintf()、fputs()和fputc()函數
int fprintf(FILE *stream, char *format, <variable-list>);
int fputs(char *string, FILE *steam);
int fputc(int ch, FILE *steam);
fprintf() 函數的返回值爲實際寫入文件中的字罕個數(字節數)。如果寫錯誤, 則返回一個負數,
fputs()函數返回0時表明將string指針所指的字符串寫入文件中的操作成功, 返回非0時, 表明寫操作失敗。
fputc()函數返回一個向文件所寫字符的值, 此時寫操作成功, 否則返回EOF(文件結束結束其值爲-1, 在stdio.h中定義)表示寫操作錯誤。
fprintf( ) 函數中格式化的規定與printf( ) 函數相同, 所不同的只是fprintf()函數是向文件中寫入。而printf()是向屏幕輸出。
下面介紹一個例子, 運行後產後一個test.dat的文件。
例11:
#include<stdio.h>
main()
{
char *s="That's good news"); /*定義字符串指針並初始化*/
int i=617; /*定義整型變量並初始化*/
FILE *fp; /*定義文件指針*/
fp=fopne("test.dat", "w"); /*建立一個文字文件只寫*/
fputs("Your score of TOEFLis", fp);/*向所建文件寫入一串字符*/
fputc(':', fp); /*向所建文件寫冒號:*/
fprintf(fp, "%d\n", i); /*向所建文件寫一整型數*/
fprintf(fp, "%s", s); /*向所建文件寫一字符串*/
fclose(fp); /*關閉文件*/
}
用DOS的TYPE命令顯示TEST.DAT的內容如下所示:
屏幕顯示
Your score of TOEFL is: 617
That's good news
2. 文件的順序讀操作函數
fscanf()、fgets()和fgetc()函數
函數fscanf()、fgets()和fgetc()均爲文件的順序讀操作函數, 其調用格式
如下:
int fscanf(FILE *stream, char *format, <address-list>);
char fgets(char *string, int n, FILE *steam);
int fgetc(FILE *steam);
fscanf()函數的用法與scanf()函數相似, 只是它是從文件中讀到信息。fscanf()函數的返回值爲EOF(即-1), 表明讀錯誤, 否則讀數據成功。
fgets()函數從文件中讀取至多n-1個字符(n用來指定字符數), 並把它們放入string指向的字符串中, 在讀入之後自動向字符串未尾加一個空字符, 讀成功返回string指針, 失敗返回一個空指針。
fgetc()函數返回文件當前位置的一個字符, 讀錯誤時返回EOF。
下面程序讀取例11產生的test.dat文件, 並將讀出的結果顯示在屏幕上。
例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()相同。