再讀標(biāo)準(zhǔn)IO中的讀函數(shù)
作者:王姍姍,華清遠(yuǎn)見(jiàn)嵌入式學(xué)院講師。
在標(biāo)準(zhǔn)IO中,對(duì)于文件的讀寫(xiě)無(wú)非三種方式:
(1) 每次一個(gè)字符的I / O。
(2) 每次一行的I / O。
(3) 直接I / O。
似乎這些函數(shù)看起來(lái)很簡(jiǎn)單,但只有理解了這些函數(shù),你才知道該如何使用,例如怎樣來(lái)判斷讀取的文件是否結(jié)束,怎樣來(lái)統(tǒng)計(jì)文件的行數(shù),等等。
下面先來(lái)回顧下讀取文件的三種方式,然后再舉例看下怎樣使用。
一次讀或?qū)懸粋€(gè)字符,使用getc、fgetc或getchar。如果流是帶緩存的,則標(biāo)準(zhǔn)I / O函數(shù)處理所有緩存。三個(gè)函數(shù)原型如下:
#include <stdio.h>
int getc(FILE *fp) ;
int fgetc(FILE *fp) ;
int getchar(void);
三個(gè)函數(shù)的返回:若成功則為下一個(gè)字符,若已處文件尾端或出錯(cuò)則為E O F。強(qiáng)調(diào)的是不管是出錯(cuò)還是到達(dá)文件尾端,這三個(gè)函數(shù)都返回同樣的值。為了區(qū)分這兩種不同的情況,必須調(diào)用ferror()或feof()。
一次讀或?qū)懸恍?,使用f g e t s和gets。兩個(gè)函數(shù)原型如下:
#include <stdio.h>
Char *gets(char *s) ;
Char *fgets(char *s,int size,FILE *steam)
兩個(gè)函數(shù)返回:若成功則為buf,若已處文件尾端或出錯(cuò)則為null。這兩個(gè)函數(shù)都指定了緩存地址,讀入的行將送入其中。gets()從標(biāo)準(zhǔn)輸入讀,而fgets()則從指定的流讀。
對(duì)于fgets(),必須指定緩存的長(zhǎng)度為n。此函數(shù)一直讀到下一個(gè)新行符為止,但是不超過(guò)n-1個(gè)字符,讀入的字符被送入緩存,該緩存以null字符結(jié)尾。如若該行,包括最后一個(gè)新行符的字符數(shù)超過(guò)n-1,則只返回一個(gè)不完整的行,而且緩存總是以null字符結(jié)尾。對(duì)fgets()的下一次調(diào)用會(huì)繼續(xù)讀該行。
直接I / O使用的是fread。每次I / O操作讀某種數(shù)量的對(duì)象,而每個(gè)對(duì)象具有指定的長(zhǎng)度。這個(gè)函數(shù)常用于從二進(jìn)制文件中讀一個(gè)結(jié)構(gòu)。原型如下:
int fread(void * ptr,int objsize,int objnum,FILE *fp);
這個(gè)函數(shù)返回:讀的對(duì)象數(shù)。fputs()在遇到null字節(jié)時(shí)就停止,而在結(jié)構(gòu)中可能含有null字節(jié),所以不能使用每次一行函數(shù)實(shí)現(xiàn)這種要求,而fread能使我們一次讀整個(gè)結(jié)構(gòu)。
下面看下如何利用上面提到的標(biāo)準(zhǔn)IO讀函數(shù)來(lái)判斷你讀取的文件是否結(jié)束以及如何來(lái)統(tǒng)計(jì)文件的行數(shù)。
1、判斷文件結(jié)束有三種方式:
a) 當(dāng)用fgetc(src)返回的int值為EOF時(shí),讀到文件的末尾
b) 當(dāng)用fgets(p,1024,src)返回的指針為NULL時(shí),讀到文件的末尾
c) 當(dāng)用fread(s,1,20,src)返回的個(gè)數(shù)小于20時(shí),讀到文件的末尾
實(shí)現(xiàn)起來(lái)簡(jiǎn)單明了,就不再細(xì)說(shuō)。
2、統(tǒng)計(jì)文件的行數(shù)
a) 當(dāng)用fgetc(src)返回的int值為’n’時(shí)行數(shù)加一
b) 當(dāng)用fgets(p,1024,src)返回的指針不為NULL時(shí),你的緩沖即提供的p[1024]已經(jīng)讀到字符,這個(gè)時(shí)候?qū)嶋H上有兩種情況,一種你讀進(jìn)來(lái)的字符長(zhǎng)度小于1024個(gè),這個(gè)時(shí)候一行結(jié)束,行數(shù)加一,另一種比較復(fù)雜,由于受自己定義的緩沖的限制,一次只能讀進(jìn)來(lái)1023個(gè),如果恰好這個(gè)時(shí)候你剛好讀完一行,那p[]1022]里存放的一定’n’,如果不是,那你的一行還一定沒(méi)有結(jié)束了。
程序如下:
Int n;
A) While(fgetc(src)==’n’)
N++;
B) while(fgets(p,1024,src)!=NULL)
{
If(strlen(p)<1024||p[1022]==’n’)
N++;
}
“本文由華清遠(yuǎn)見(jiàn)http://www.embedu.org/index.htm提供”
來(lái)源:華清遠(yuǎn)見(jiàn)0次