您好,登錄后才能下訂單哦!
C++ 中eof()函數(shù)如何使用,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。
在使用C/C++讀文件的時候,一定都使用過C++ eof()函數(shù)來判斷文件是否為空或者是否讀到文件結(jié)尾了,也會在使用這個函數(shù)的過程中遇到一些問題,如不能準確的判斷是否為空或者是否到了文件尾,以至于有些人可能還會懷疑這個函數(shù)是不是本身在設計上就有問題。
先來看看如下這段代碼:
#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; }
如果test.txt不存在,程序會形成死循環(huán),fin.eof()永遠返回false,如果test.txt為空,程序打印出一個x字符,當test.txt中存在一字符串“abcd”且沒有換行時,程序打印出“abcdd”,當存在以上字符串并且有一新的空行時,程序打印出“abcd”加上一空行。
這種現(xiàn)象可能讓很多人很迷惑,程序運行的結(jié)果似乎很不穩(wěn)定,時對時錯。使用binary模式讀時結(jié)果一樣。在這里,大家可能有一個誤區(qū),認為eof()返回true時是讀到文件的***一個字符,其實不然,C++ eof()函數(shù)返回true時是讀到文件結(jié)束符0xFF,而文件結(jié)束符是***一個字符的下一個字符。
因此,當讀到***一個字符時,程序會多讀一次(編譯器會讓指針停留在***一個字符那里,然后重復讀取一次,這也就是就上面***一個字符會輸出兩次的原因。至于是不是所有的編譯器都這樣處理我就不太清楚了,我使用的VC6,VC8似乎都是這樣的)
問題出來了,就要找出對應的解決之道,要解決以上的問題,只需要調(diào)整一下條件語句即可:
fin.peek() == EOF 或 fin.get(ch)
再來看一下另外一種情況:
#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下編譯運行,發(fā)現(xiàn),當文件結(jié)尾沒有空行時,結(jié)果正確,當結(jié)尾有空行時,***一個字符串將被重復輸出一次, 而VC6的情況則有所不同,沒有重復輸出,但輸出了一個空行。
因此,為了保證在不同的編譯器下得到一致的我們期望的結(jié)果,將條件語句做一下修改:
fin >> str
綜上對C++ eof()函數(shù)的描述,我們可以得到以下結(jié)論:
1. 判斷文件是否為空時使用peek函數(shù),若peek返回EOF則文件為空;
2. 讀取文件過程中,讀取非char型時,使用peek判斷文件尾將不再適用,循環(huán)判斷條件應改用>>操作符進行讀取,若讀入char型緩沖區(qū),peek函數(shù)會表現(xiàn)得很好。
看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進一步的了解或閱讀更多相關(guān)文章,請關(guān)注億速云行業(yè)資訊頻道,感謝您對億速云的支持。
免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。