函數fgets和fputs、fread和fwrite、fscanf和fprintf用法小結
字元串讀寫函數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>
#include<conio.h>
#include<stdlib.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中追加一個字元串。
char ch,st[20];
if((fp=fopen("string","at+"))==NULL)
printf("input a string:\n");
scanf("%s",st);
fputs(st,fp);
rewind(fp);
ch=fgetc(fp);
while(ch!=EOF)
putchar(ch);
printf("\n");
本例要求在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]從鍵盤輸入兩個學生資料,寫入一個檔案中, 再讀出這兩個學生的資料顯示在螢幕上。
struct stu
char name[10];
int num;
int age;
char addr[15];
}boya[2],boyb[2],*pp,*qq;
char ch;
int i;
pp=boya;
qq=boyb;
if((fp=fopen("stu_list","wb+"))==NULL)
printf("\ninput data\n");
for(i=0;i<2;i++,pp++)
scanf("%s%d%d%s",pp->name,&pp->num,&pp->age,pp->addr);
fwrite(pp,sizeof(struct stu),2,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);
本例程式定義了一個結構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]
fprintf(fp,"%s %d %d %s\n",pp->name,pp->num,pp->age,pp->
addr);
fscanf(fp,"%s %d %d %s\n",qq->name,&qq->num,&qq->age,qq->addr);
printf("%s\t%5d %7d %s\n",qq->name,qq->num, qq->age,
qq->addr);
與例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中讀出第二個學生的資料。
}boy,*qq;
int i=1;
qq=&boy;
if((fp=fopen("stu_list","rb"))==NULL)
fseek(fp,i*sizeof(struct stu),0);
fread(qq,sizeof(struct stu),1,fp);
printf("%s\t%5d %7d %s\n",qq->name,qq->num,qq->age,
檔案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相相容而說明的一些常量,包括浮點和雙精度值的範圍。
C語言中如何讀取以逗号分割的檔案中的資料
要有逗号,就全加逗号,
2,1,34,2,3,
1,400,5,
3,5,6,7,
讀:
for (j=0;j<NR;j++)
for (i=0;i<NC;i++)
fscanf(fin,"%d,",a[j][i]);
-----------------------------
如果行尾無逗号,每行資料的個數應相等:
2,1,34,2
3,1,400,5
for (j=0;j<NR;j++){
for (i=0;i<NC-1;i++) fscanf(fin,"%d,",a[j][i]);
fscanf(fin,"%d,",a[j][NC-1]);
随機有逗号,随機行長
2,1,34,2,3
1,400,5
3,5,6,7
用fgets讀入,計算行長,按字元循環找逗号,并用空白代替逗号,再用sscanf 讀入資料
----------------------------
不知道總個數。一是開大數組。
二是預讀,數出個數,開數組,
rewind(fin);
再從頭輸入。
常用基本參數對照:
%d:讀入一個十進制整數.
%i :讀入十進制,八進制,十六進制整數,與%d類似,但是在編譯時通過資料前置來區分進制,如加入“0x”則是十六進制,加入“0”則為八進制。例如串“031”使用%d時會被算作31,但是使用%i時會算作25.
%u:讀入一個無符号十進制整數.
%f %F %g %G : 用來輸入實數,可以用小數形式或指數形式輸入.
%x %X: 讀入十六進制整數.
%o: 讀入八進制整數.
%s : 讀入一個字元串,遇空格結束。
%c : 讀入一個字元。無法讀入空值。空格可以被讀入。
附加格式說明字元表修飾符說明
L/l 長度修飾符 輸入"長"資料
h 長度修飾符 輸入"短"資料