字串讀寫函式fgets和fputs
一、讀字串函式fgets函式的功能是從指定的檔案中讀一個字串到字元陣列中,函式呼叫的形式為: fgets(字元陣列名,n,檔案指標); 其中的n是一個正整數。表示從檔案中讀出的字串不超過 n-1個字元。在讀入的最後一個字元後加上串結束標誌'\0'。例如:fgets(str,n,fp);的意義是從fp所指的檔案中讀出n-1個字元送入 字元陣列str中。
[例10.4]從e10_1.c檔案中讀入一個含10個字元的字串。
#include<stdio.h>
main()
{
FILE *fp;
char str[11];
if((fp=fopen("e10_1.c","rt"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
fgets(str,11,fp);
printf("%s",str);
fclose(fp);
}
本例定義了一個字元陣列str共11個位元組,在以讀文字檔案方式開啟檔案e101.c後,從中讀出10個字元送入str陣列,在陣列最後一個單元內將加上'\0',然後在螢幕上顯示輸出str陣列。輸出的十個字元正是例10.1程式的前十個字元。
對fgets函式有兩點說明:
1. 在讀出n-1個字元之前,如遇到了換行符或EOF,則讀出結束。
2. fgets函式也有返回值,其返回值是字元陣列的首地址。
二、寫字串函式fputs
fputs函式的功能是向指定的檔案寫入一個字串,其呼叫形式為: fputs(字串,檔案指標) 其中字串可以是字串常量,也可以是字元陣列名, 或指標 變數,例如:
fputs(“abcd“,fp);
其意義是把字串“abcd”寫入fp所指的檔案之中。[例10.5]在例10.2中建立的檔案string中追加一個字串。
#include<stdio.h>
main()
{
FILE *fp;
char ch,st[20];
if((fp=fopen("string","at+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("input a string:\n");
scanf("%s",st);
fputs(st,fp);
rewind(fp);
ch=fgetc(fp);
while(ch!=EOF)
{
putchar(ch);
ch=fgetc(fp);
}
printf("\n");
fclose(fp);
}
本例要求在string檔案末加寫字串,因此,在程式第6行以追加讀寫文字檔案的方式開啟檔案string 。 然後輸入字串, 並用fputs函式把該串寫入檔案string。在程式15行用rewind函式把檔案內部位置指標移到檔案首。 再進入迴圈逐個顯示當前檔案中的全部內容。
資料塊讀寫函式fread和fwrite
C語言還提供了用於整塊資料的讀寫函式。 可用來讀寫一組資料,如一個陣列元素,一個結構變數的值等。讀資料塊函式呼叫的一般形式為: fread(buffer,size,count,fp); 寫資料塊函式呼叫的一般形式為: fwrite(buffer,size,count,fp); 其中buffer是一個指標,在fread函式中,它表示存放輸入資料的首地址。在fwrite函式中,它表示存放輸出資料的首地址。 size 表示資料塊的位元組數。count 表示要讀寫的資料塊塊數。fp 表示檔案指標。
例如:
fread(fa,4,5,fp); 其意義是從fp所指的檔案中,每次讀4個位元組(一個實數)送入實陣列fa中,連續讀5次,即讀5個實數到fa中。
[例10.6]從鍵盤輸入兩個學生資料,寫入一個檔案中, 再讀出這兩個學生的資料顯示在螢幕上。
#include<stdio.h>
struct stu
{
char name[10];
int num;
int age;
char addr[15];
}boya[2],boyb[2],*pp,*qq;
main()
{
FILE *fp;
char ch;
int i;
pp=boya;
qq=boyb;
if((fp=fopen("stu_list","wb+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("\ninput data\n");
for(i=0;i<2;i++,pp++)
scanf("%s%d%d%s",pp->name,&pp->num,&pp->age,pp->addr);
pp=boya;
fwrite(pp,sizeof(struct stu),2,fp);
rewind(fp);
fread(qq,sizeof(struct stu),2,fp);
printf("\n\nname\tnumber age addr\n");
for(i=0;i<2;i++,qq++)
printf("%s\t%5d%7d%s\n",qq->name,qq->num,qq->age,qq->addr);
fclose(fp);
}
本例程式定義了一個結構stu,說明了兩個結構陣列boya和 boyb以及兩個結構指標變數pp和qq。pp指向boya,qq指向boyb。程式第16行以讀寫方式開啟二進位制檔案“stu_list”,輸入二個學 生資料之後,寫入該檔案中, 然後把檔案內部位置指標移到檔案首,讀出兩塊學生資料後,在螢幕上顯示。
格式化讀寫函式fscanf和fprintf
fscanf函式,fprintf函式與前面使用的scanf和printf 函式的功能相似,都是格式化讀寫函式。 兩者的區別在於 fscanf 函式和fprintf函式的讀寫物件不是鍵盤和顯示器,而是磁碟檔案。這兩個函式的呼叫格式為: fscanf(檔案指標,格式字串,輸入表列); fprintf(檔案指標,格式字串,輸出表列); 例如:
fscanf(fp,"%d%s",&i,s);
fprintf(fp,"%d%c",j,ch);
用fscanf和fprintf函式也可以完成例10.6的問題。修改後的程式如例10.7所示。
[例10.7]
#include<stdio.h>
struct stu
{
char name[10];
int num;
int age;
char addr[15];
}boya[2],boyb[2],*pp,*qq;
main()
{
FILE *fp;
char ch;
int i;
pp=boya;
qq=boyb;
if((fp=fopen("stu_list","wb+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("\ninput data\n");
for(i=0;i<2;i++,pp++)
scanf("%s%d%d%s",pp->name,&pp->num,&pp->age,pp->addr);
pp=boya;
for(i=0;i<2;i++,pp++)
fprintf(fp,"%s %d %d %s\n",pp->name,pp->num,pp->age,pp->
addr);
rewind(fp);
for(i=0;i<2;i++,qq++)
fscanf(fp,"%s %d %d %s\n",qq->name,&qq->num,&qq->age,qq->addr);
printf("\n\nname\tnumber age addr\n");
qq=boyb;
for(i=0;i<2;i++,qq++)
printf("%s\t%5d %7d %s\n",qq->name,qq->num, qq->age,
qq->addr);
fclose(fp);
}
與例10.6相比,本程式中fscanf和fprintf函式每次只能讀寫一個結構陣列元素,因此採用了迴圈語句來讀寫全部陣列元素。 還要注意指標變數pp,qq由於迴圈改變了它們的值,因此在程式的25和32行分別對它們重新賦予了陣列的首地址。
檔案的隨機讀寫
前面介紹的對檔案的讀寫方式都是順序讀寫, 即讀寫檔案只能從頭開始,順序讀寫各個資料。 但在實際問題中常要求只讀寫檔案中某一指定的部分。 為了解決這個問題可移動檔案內部的位置指標到需要讀寫的位置,再進行讀寫,這種讀寫稱為隨機讀寫。 實現隨機讀寫的關鍵是要按要求移動位置指標,這稱為檔案的定位。檔案定位移動檔案內部位置指標的函式主要有兩個, 即 rewind 函式和fseek函式。
rewind函式前面已多次使用過,其呼叫形式為: rewind(檔案指標); 它的功能是把檔案內部的位置指標移到檔案首。 下面主要介紹
fseek函式。
fseek函式用來移動檔案內部位置指標,其呼叫形式為: fseek(檔案指標,位移量,起始點); 其中:“檔案指標”指向被移動的檔案。 “位移量”表示移動的位元組數,要求位移量是long型資料,以便在檔案長度大於64KB 時不會出錯。當用常量表示位移量時,要求加字尾“L”。“起始點”表示從何處開始計算位移量,規定的起始點有三種:檔案首,當前位置和檔案尾。
其表示方法如表10.2。
起始點 表示符號 數字表示
──────────────────────────
檔案首 SEEK—SET 0
當前位置 SEEK—CUR 1
檔案末尾 SEEK—END 2
例如:
fseek(fp,100L,0);其意義是把位置指標移到離檔案首100個位元組處。還要說明的是fseek函式一般用於二進位制檔案。在文字檔案中由 於要進行轉換,故往往計算的位置會出現錯誤。檔案的隨機讀寫在移動位置指標之後, 即可用前面介紹的任一種讀寫函式進行讀寫。由於一般是讀寫一個資料據塊,因此常用fread和fwrite函式。下面用例題來說明檔案的隨機讀寫。
[例10.8]在學生檔案stu list中讀出第二個學生的資料。
#include<stdio.h>
struct stu
{
char name[10];
int num;
int age;
char addr[15];
}boy,*qq;
main()
{
FILE *fp;
char ch;
int i=1;
qq=&boy;
if((fp=fopen("stu_list","rb"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
rewind(fp);
fseek(fp,i*sizeof(struct stu),0);
fread(qq,sizeof(struct stu),1,fp);
printf("\n\nname\tnumber age addr\n");
printf("%s\t%5d %7d %s\n",qq->name,qq->num,qq->age,
qq->addr);
}
檔案stu_list已由例10.6的程式建立,本程式用隨機讀出的方法讀出第二個學生的資料。程式中定義boy為stu型別變數,qq為指向boy 的指標。以讀二進位制檔案方式開啟檔案,程式第22行移動檔案位置指標。其中的i值為1,表示從檔案頭開始,移動一個stu型別的長度, 然後再讀出的資料即為第二個學生的資料。
檔案檢測函式
C語言中常用的檔案檢測函式有以下幾個。
一、檔案結束檢測函式feof函式呼叫格式: feof(檔案指標);
功能:判斷檔案是否處於檔案結束位置,如檔案結束,則返回值為1,否則為0。
二、讀寫檔案出錯檢測函式ferror函式呼叫格式: ferror(檔案指標);
功能:檢查檔案在用各種輸入輸出函式進行讀寫時是否出錯。 如ferror返回值為0表示未出錯,否則表示有錯。
三、檔案出錯標誌和檔案結束標誌置0函式clearerr函式呼叫格式: clearerr(檔案指標);
功能:本函式用於清除出錯標誌和檔案結束標誌,使它們為0值。
C庫檔案
C系統提供了豐富的系統檔案,稱為庫檔案,C的庫檔案分為兩類,一類是副檔名為".h"的檔案,稱為標頭檔案, 在前面的包含命令中我們已多次使用過。在".h"檔案中包含了常量定義、 型別定義、巨集定義、函式原型以及各種編譯選擇設定等資訊。另一類是函式庫,包括了各種函式的目的碼,供使用者在程式中呼叫。 通常在程式中呼叫一個庫函式時,要在呼叫之前包含該函式原型所在的".h" 檔案。
在附錄中給出了全部庫函式。
ALLOC.H 說明記憶體管理函式(分配、釋放等)。
ASSERT.H 定義 assert除錯巨集。
BIOS.H 說明呼叫IBM—PC ROM BIOS子程式的各個函式。
CONIO.H 說明呼叫DOS控制檯I/O子程式的各個函式。
CTYPE.H 包含有關字元分類及轉換的名類資訊(如 isalpha和toascii等)。
DIR.H 包含有關目錄和路徑的結構、巨集定義和函式。
DOS.H 定義和說明MSDOS和8086呼叫的一些常量和函式。
ERRON.H 定義錯誤程式碼的助記符。
FCNTL.H 定義在與open庫子程式連線時的符號常量。
FLOAT.H 包含有關浮點運算的一些引數和函式。
GRAPHICS.H 說明有關圖形功能的各個函式,圖形錯誤程式碼的常量定義,正對不同驅動程式的各種顏色值,及函式用到的一些特殊結構。
IO.H 包含低階I/O子程式的結構和說明。
LIMIT.H 包含各環境引數、編譯時間限制、數的範圍等資訊。
MATH.H 說明數學運算函式,還定了 HUGE VAL 巨集, 說明了matherr和matherr子程式用到的特殊結構。
MEM.H 說明一些記憶體操作函式(其中大多數也在STRING.H 中說明)。
PROCESS.H 說明程式管理的各個函式,spawn…和EXEC …函式的結構說明。
SETJMP.H 定義longjmp和setjmp函式用到的jmp buf型別, 說明這兩個函式。
SHARE.H 定義檔案共享函式的引數。
SIGNAL.H 定義SIG[ZZ(Z] [ZZ)]IGN和SIG[ZZ(Z] [ZZ)]DFL常量,說明rajse和signal兩個函式。
STDARG.H 定義讀函式參數列的巨集。(如vprintf,vscarf函式)。
STDDEF.H 定義一些公共資料型別和巨集。
STDIO.H 定義Kernighan和Ritchie在Unix System V 中定義的標準和擴充套件的型別和巨集。還定義標準I/O 預定義流:stdin,stdout和stderr,說明 I/O流子程式。
STDLIB.H 說明一些常用的子程式:轉換子程式、搜尋/ 排序子程式等。
STRING.H 說明一些串操作和記憶體操作函式。
SYS\STAT.H 定義在開啟和建立檔案時用到的一些符號常量。
SYS\TYPES.H 說明ftime函式和timeb結構。
SYS\TIME.H 定義時間的型別time[ZZ(Z] [ZZ)]t。
TIME.H 定義時間轉換子程式asctime、localtime和gmtime的結構,ctime、 difftime、 gmtime、 localtime和stime用到的型別,並提供這些函式的原型。
VALUE.H 定義一些重要常量, 包括依賴於機器硬體的和為與Unix System V相相容而說明的一些常量,包括浮點和雙精度值的範圍。