?? 12.2.6 文件的讀取.txt
字號(hào):
12.2.6 文件的讀取
對(duì)于文件的讀取,首先也需要打開(kāi)文件,即需要調(diào)用 fopen函數(shù)得到指向 FILE類(lèi)型的指針,然后如
果要讀取文件的內(nèi)容,應(yīng)該調(diào)用fread函數(shù)。該函數(shù)的聲明形式如下所示:
size_t fread( void *buffer, size_t size, size_t count , F工LE *stream );
可以看到,與fwrite函數(shù)一樣, fread函數(shù)也有四個(gè)參數(shù),并且各參數(shù)的含義與fwrite函數(shù)的相應(yīng)
參數(shù)含義相同,只是這時(shí)第一個(gè)參數(shù): buffer是指向用來(lái)存放數(shù)據(jù)的緩沖區(qū)的指針。
下面,我們就在File程序CFileView類(lèi)的OnFileRead函數(shù)中添加如例 12-5所示的代碼,利用C語(yǔ)言提
供的函數(shù)實(shí)現(xiàn)文件的讀取操作。
例12-5
'
void CFileView: :OnFileRead()
FILE *pF工le=fopen ( "1 . txt " , "r" ) ;
char ch[100] ;
fread(ch , 1 , 100 , pFile) ;
fclose (pFile) ;
MessageBox (ch) ;
在上述如例 12-5所示代碼中,首先調(diào)用fopen函數(shù)得到想要讀取其內(nèi)容的文件結(jié)構(gòu)指針,本例將讀
取上面寫(xiě)入操作使用的文件: l.txt。因?yàn)橹皇且x取文件中的數(shù)據(jù),所以我們使用"r"的方式打開(kāi)。
接著,定義了一個(gè) 100個(gè)字符的字符數(shù)組,用來(lái)接收讀取的數(shù)據(jù)。然后調(diào)用fread函數(shù)讀取剛才寫(xiě)入
文件操作時(shí)使用的文件: l.txt中的內(nèi)容。同樣,在文件讀取操作完成之后,可以使用 fclose函數(shù)
關(guān)閉文件。最后,調(diào)用MessageBox函數(shù)將讀取到的數(shù)據(jù)顯示出來(lái)。
Build井運(yùn)行 File程序,利用相應(yīng)菜單命令先寫(xiě)入文件,再讀取文件,這時(shí)在程序顯示的消息框中
就可以看到讀取到的數(shù)據(jù),如圖 12.6所示。
圖 12.6讀取文件數(shù)據(jù)后顯示結(jié)果不正確
‘~~ 1443
第12
可是,我們發(fā)現(xiàn)該消息框顯示的文字中,開(kāi)始部分是我們程序中寫(xiě)入的數(shù)據(jù): "ftp://
www.suruin-org",但后面顯示了一串亂碼。這里再一次強(qiáng)調(diào),在C語(yǔ)言中,字符串是以"\。"符號(hào)結(jié)
束的。在顯示字符串時(shí),系統(tǒng)會(huì)尋找"\。"字符。而這里定義的1∞個(gè)元素的字符數(shù)組: ch,它里面
的數(shù)據(jù)是隨機(jī)的,可能有許多數(shù)據(jù)都是不可讀的,系統(tǒng)會(huì)從該字符數(shù)組的開(kāi)始地址依次往后查找,
由于在其中沒(méi)有找到氣。"字符,于是它就把這些不可讀的數(shù)據(jù)顯示出來(lái)了,結(jié)果就是我們所看到的
亂碼。因此,我們應(yīng)該在讀取到的數(shù)據(jù)之后添加氣。"字符,以便作為字符串的結(jié)尾。
有多種方法可以向一個(gè)字符串添加"飛。"字符,以表示該字符串的結(jié)束。
·第一種方法
可以在寫(xiě)入數(shù)據(jù)時(shí)多寫(xiě)入一個(gè)字節(jié),例如上面寫(xiě)入操作向l.txt文件中寫(xiě)入的字符串 ''http://www.
sunxin.org"是21個(gè)字節(jié),可以在寫(xiě)入時(shí)在其后再添加一個(gè)字節(jié),其內(nèi)容設(shè)置為"飛。",即這時(shí)的寫(xiě)
入文件代碼如例 12-6所示。
例 12-6
void CFileView : : OnFileWrite()
FILE *pFile=fopen (" 1 . txt" , "w") ;
char buf[22) = ''http://www . sunxin. org" ;
buf [21) = '\o ' ;
fwrite(buf , 1 , 22 ,pFile) ;
fclose (pFile) ;
運(yùn)行File程序,單擊【文件操作飛寫(xiě)入文件】菜單項(xiàng)后,可以發(fā)現(xiàn)生成的l.txt文件大小是22個(gè)字節(jié)。
讀者可以以二進(jìn)制方試打開(kāi)該文件,可以看到最后一個(gè)數(shù)據(jù)就是: 0。讀者應(yīng)注
意:"飛。"就是 0。在讀取文件數(shù)據(jù)時(shí),"\。"字符也將被讀取,當(dāng)把讀取到的數(shù)據(jù)存入字符數(shù)組后,
就可以根據(jù)其中的"\。"字符判圖 12.7讀取文件數(shù)據(jù)后斷出字符串的結(jié)尾。這時(shí),再單擊【文件操
作L讀取文件】菜單顯示的正確結(jié)果,在彈出的消息框中就可以看到正確的內(nèi)容了,結(jié)果如圖
12.7所示。
這種方式的缺點(diǎn)是增加了文件大小,雖然只是多寫(xiě)了一個(gè)字節(jié),但是從文件大小的精確度角度來(lái)說(shuō),
這種方式不是很好。
·第二種方法
在定義字符數(shù)組之后,利用C語(yǔ)言中的memset函數(shù)將這個(gè)字符數(shù)組中的所有數(shù)據(jù)都設(shè)置為o(代碼如例
12-7所示〕。然后在程序中讀取文件時(shí),讀取到多少數(shù)據(jù)就會(huì)在該字符數(shù)組中存放多少數(shù)據(jù),字符
數(shù)組中剩下的數(shù)據(jù)全是0。這樣,在字符串顯示時(shí)遇到"飛\0"就表示該字符串結(jié)束了,因此就能正確
地顯示數(shù)據(jù)。
例 12-7
void CFileView: : OnFileRead() F工LE *pFile=fopen("1.txt " , "r" );
char ch[100] ;
memset(ch, O, 100);
fread(ch , 1 , 100 ,pFile) ;
fclose(pFile) ;
MessageBox (ch) ;
·第二種方法
有時(shí)在讀取文件時(shí),并不知道文件的大小,這里我們定義的字符數(shù)組固定為 100個(gè)元素,如果容量
不夠怎么辦呢?所以在程序中需要獲取文件的長(zhǎng)度,然后根據(jù)這個(gè)長(zhǎng)度來(lái)分配存放該數(shù)據(jù)的內(nèi)存。可
以利用C語(yǔ)言中的ftell函數(shù)來(lái)得到文件的長(zhǎng)度。 ftell函數(shù)將返回文件指針當(dāng)前的位置。因此,可
以先利用 fseek函數(shù)將文件指針移動(dòng)到文件的結(jié)尾處,然后利用ftell函數(shù)就可以得到文件指針當(dāng)前
的位置,也就是文件的長(zhǎng)度。具體代碼如例 12-8所示。
例12-8
1. void CFileView::OnFileRead()
2. {
3. FILE *pFile=fopen( "l .txt " , "r") ;
4. char *pBuf;
5. fseek(pFile , O, SEEK_END) ;
6. int len=ftell(pFile) ;
7 . pBuf=new char[len+1] ;
8. fread(pBuf,1,len,pFile) ;
9. pBuf[len]=0 ;
10. fclose (pFile) ;
11. MessageBox(pBuf) ;
12. }
在上述例 12-8所示代碼中,得到文件長(zhǎng)度 Clen)之后,分配內(nèi)存以便用來(lái)保存讀取到的數(shù)據(jù),這時(shí)
應(yīng)該多分配一個(gè)字節(jié),用來(lái)存放表示字符串結(jié)尾的"舊"字符,然后就可以讀取文件的內(nèi)容了,并將
數(shù)組的最后一個(gè)元素設(shè)置為o (例12-8所示代碼的第9行代碼〉。對(duì)于數(shù)組來(lái)說(shuō),它的索引從O開(kāi)始,
因此對(duì)于元素個(gè)數(shù)為 len+1的數(shù)組來(lái)說(shuō),索引為len的元素就是它的最后一個(gè)元素。接下來(lái)就可以調(diào)
用 fclose函數(shù)關(guān)閉文件。最后調(diào)用 MessageBox函數(shù)顯示讀取到的數(shù)據(jù)。
再次運(yùn)行File程序,單擊【文件操作\讀取文件】菜單項(xiàng),你會(huì)發(fā)現(xiàn)這時(shí)程序顯示的消息框中顯示的
數(shù)據(jù)又出現(xiàn)了亂碼。我們對(duì)上述例12-8所示代碼進(jìn)行分析,在我們將文件指針移動(dòng)到文件結(jié)尾后(上
述如例 12-8所示代碼的第5行),再次讀取文件時(shí)(上述如例 12-8所示代碼的第 8行),就是從文件
指針指向的下一個(gè)寫(xiě)入位置處開(kāi)始讀取,而此時(shí)文件指針已經(jīng)到了文件結(jié)尾,當(dāng)然是讀取不到數(shù)據(jù)
的。這是在對(duì)文件進(jìn)行操作時(shí),移動(dòng)文件指針后經(jīng)常會(huì)犯的一個(gè)錯(cuò)誤。在這里,我們應(yīng)該在讀取之
前將文件指針再次移動(dòng)到文件開(kāi)始處,當(dāng)然這可以通過(guò)fseek函數(shù)來(lái)實(shí)現(xiàn),但這里我們利用另一個(gè)函
數(shù): rewind來(lái)實(shí)現(xiàn)這一功能,該函數(shù)的功能是將文件指針重新放置到文件的開(kāi)始處。 rewind函數(shù)只
有一個(gè)參數(shù),就是指
向 FILE結(jié)構(gòu)體的指針。因此在上述讀取代碼(上述如例 12-8所示代碼的第 8行)之前,添加下面這
條語(yǔ)句,將文件指針重新放置到文件開(kāi)始處。
rewind (pFile) ;
再次運(yùn)行 File程序,單擊【文件操作飛讀取文件】菜單項(xiàng),這時(shí)可以發(fā)現(xiàn)程序顯示的消息框中顯示
的結(jié)果正確了。這里,再次提醒讀者在使用 C語(yǔ)言函數(shù)對(duì)文件進(jìn)行讀取操作時(shí),應(yīng)注意以下幾點(diǎn):
·
在讀取文件數(shù)據(jù)時(shí),如果是字符數(shù)據(jù),通常在定義用來(lái)保存該數(shù)據(jù)的字符數(shù)組時(shí),在字符數(shù)據(jù)個(gè)數(shù)
的基礎(chǔ)上,多分配一個(gè)字節(jié),以存放表示字符串結(jié)尾的字符 :"\。"。
·
在讀取文件內(nèi)容時(shí),應(yīng)正確地設(shè)置文件指針的位置。
?? 快捷鍵說(shuō)明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號(hào)
Ctrl + =
減小字號(hào)
Ctrl + -