C++ eof()函數(shù)相關(guān)應(yīng)用技巧分享
C++編程語(yǔ)言中的很多功能在我們的實(shí)際應(yīng)用中起著非常大的作用。比如在對(duì)文件文本的操作上,就可以用多種方式來(lái)實(shí)現(xiàn)。在這里我們介紹的C++ eof()函數(shù)就是其中一個(gè)比較常用的基本函數(shù)。
在使用C/C++讀文件的時(shí)候,一定都使用過(guò)C++ eof()函數(shù)來(lái)判斷文件是否為空或者是否讀到文件結(jié)尾了,也會(huì)在使用這個(gè)函數(shù)的過(guò)程中遇到一些問(wèn)題,如不能準(zhǔn)確的判斷是否為空或者是否到了文件尾,以至于有些人可能還會(huì)懷疑這個(gè)函數(shù)是不是本身在設(shè)計(jì)上就有問(wèn)題。
先來(lái)看看如下這段代碼:
- #include < iostream>
- #include < fstream>
- using namespace std;
- int main()
- {
- char ch = 'x';
- ifstream fin("test.txt" /*, ios::binary*/);
- if (fin.eof())
- {
- cout < < "file is empty."< < endl;
- return 0;
- }
- while (!fin.eof())
- {
- fin.get(ch);
- cout < < ch;
- }
- system("pause");
- return 0;
- }
編譯并運(yùn)行以上代碼,#t#
如果test.txt不存在,程序會(huì)形成死循環(huán),fin.eof()永遠(yuǎn)返回false,如果test.txt為空,程序打印出一個(gè)x字符,當(dāng)test.txt中存在一字符串“abcd”且沒(méi)有換行時(shí),程序打印出“abcdd”,當(dāng)存在以上字符串并且有一新的空行時(shí),程序打印出“abcd”加上一空行。
這種現(xiàn)象可能讓很多人很迷惑,程序運(yùn)行的結(jié)果似乎很不穩(wěn)定,時(shí)對(duì)時(shí)錯(cuò)。使用binary模式讀時(shí)結(jié)果一樣。在這里,大家可能有一個(gè)誤區(qū),認(rèn)為eof()返回true時(shí)是讀到文件的***一個(gè)字符,其實(shí)不然,C++ eof()函數(shù)返回true時(shí)是讀到文件結(jié)束符0xFF,而文件結(jié)束符是***一個(gè)字符的下一個(gè)字符。
因此,當(dāng)讀到***一個(gè)字符時(shí),程序會(huì)多讀一次(編譯器會(huì)讓指針停留在***一個(gè)字符那里,然后重復(fù)讀取一次,這也就是就上面***一個(gè)字符會(huì)輸出兩次的原因。至于是不是所有的編譯器都這樣處理我就不太清楚了,我使用的VC6,VC8似乎都是這樣的)
問(wèn)題出來(lái)了,就要找出對(duì)應(yīng)的解決之道,要解決以上的問(wèn)題,只需要調(diào)整一下條件語(yǔ)句即可:
- fin.peek() == EOF 或 fin.get(ch)
再來(lái)看一下另外一種情況:
- #include < iostream>
- #include < fstream>
- #include < string>
- using namespace std;
- int main()
- {
- string str;
- ifstream fin("test.txt"/*, ios::binary*/);
- if (fin.peek() == EOF)
- {
- cout < < "file is empty."< < endl;
- return 0;
- }
- while (!fin.eof())
- {
- fin >> str;
- cout < < str;
- }
- system("pause");
- return 0;
- }
上述代碼在VC8下編譯運(yùn)行,發(fā)現(xiàn),當(dāng)文件結(jié)尾沒(méi)有空行時(shí),結(jié)果正確,當(dāng)結(jié)尾有空行時(shí),***一個(gè)字符串將被重復(fù)輸出一次, 而VC6的情況則有所不同,沒(méi)有重復(fù)輸出,但輸出了一個(gè)空行。
因此,為了保證在不同的編譯器下得到一致的我們期望的結(jié)果,將條件語(yǔ)句做一下修改:
- fin >> str
綜上對(duì)C++ eof()函數(shù)的描述,我們可以得到以下結(jié)論:
1. 判斷文件是否為空時(shí)使用peek函數(shù),若peek返回EOF則文件為空;
2. 讀取文件過(guò)程中,讀取非char型時(shí),使用peek判斷文件尾將不再適用,循環(huán)判斷條件應(yīng)改用>>操作符進(jìn)行讀取,若讀入char型緩沖區(qū),peek函數(shù)會(huì)表現(xiàn)得很好。