c++ 讀取文件 最后一行讀取了兩次

用ifstream的eof(),竟然讀到文件最后了,判斷eof還為false。網上查找資料后,終于解決這個問題。

參照文件:http://tuhao.blogbus.com/logs/21306687.html

?

在使用C/C++讀文件的時候,一定都使用過eof()這個函數來判斷文件是否為空或者是否讀到文件結尾了,也會在使用這個函數的過程中遇到一些問題,如不能準確的判斷是否為空或者是否到了文件尾,以至于有些人可能還會懷疑這個函數是不是本身在設計上就有問題。

先來看看如下這段代碼:

#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不存在,程序會形成死循環,fin.eof()永遠返回false,
如果test.txt為空,程序打印出一個x字符,
當test.txt中存在一字符串“abcd”且沒有換行時,程序打印出“abcdd”,
當存在以上字符串并且有一新的空行時,程序打印出“abcd”加上一空行。

這種現象可能讓很多人很迷惑,程序運行的結果似乎很不穩定,時對時錯。使用binary模式讀時結果一樣。在這里,大家可能有一個誤區,認為eof()返回true時是讀到文件的最后一個字符,其實不然,eof()返回true時是讀到文件結束符0xFF,而文件結束符是最后一個字符的下一個字符。如下圖所示:


?

因此,當讀到最后一個字符時,程序會多讀一次(編譯器會讓指針停留在最后一個字符那里,然后重復讀取一次,這也就是就上面最后一個字符會輸出兩次的原因。至于是不是所有的編譯器都這樣處理我就不太清楚了,我使用的VC6,VC8似乎都是這樣的)

問題出來了,就要找出對應的解決之道,要解決以上的問題,只需要調整一下條件語句即可:
????????????????????????? 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下編譯運行,發現,當文件結尾沒有空行時,結果正確,當結尾有空行時,最后一個字符串將被重復輸出一次, 而VC6的情況則有所不同,沒有重復輸出,但輸出了一個空行。

因此,為了保證在不同的編譯器下得到一致的我們期望的結果,將條件語句做一下修改:
?????????????????????????????????????????? fin?>> str??????????????????????????????????????????????????????

綜上所述,我們可以得到以下結論:
1. 判斷文件是否為空時使用peek函數,若peek返回EOF則文件為空;
2. 讀取文件過程中,讀取非char型時,使用peek判斷文件尾將不再適用,循環判斷條件應改用>>操作符進行讀取,若讀入char型緩沖區,peek函數會表現得很好。

?

peek() —— 此函數將返回輸入流文件的下一個字符,但它不移動內置指針。

轉載于:https://www.cnblogs.com/yyxayz/p/4078178.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/273531.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/273531.shtml
英文地址,請注明出處:http://en.pswp.cn/news/273531.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

java中的io系統詳解(轉)

Java 流在處理上分為字符流和字節流。字符流處理的單元為 2 個字節的 Unicode 字符&#xff0c;分別操作字符、字符數組或字符串&#xff0c;而字節流處理單元為 1 個字節&#xff0c;操作字節和字節數組。 Java 內用 Unicode 編碼存儲字符&#xff0c;字符流處理類負責將外部的…

js獲取字符串最后一個字符代碼

方法一&#xff1a;運用String對象下的charAt方法 charAt() 方法可返回指定位置的字符。 代碼如下 復制代碼 str.charAt(str.length – 1) 請注意&#xff0c;JavaScript 并沒有一種有別于字符串類型的字符數據類型&#xff0c;所以返回的字符是長度為 1 的字符串 方法二&#…

Unity3D Shader入門指南(二)

關于本系列 這是Unity3D Shader入門指南系列的第二篇&#xff0c;本系列面向的對象是新接觸Shader開發的Unity3D使用者&#xff0c;因為我本身自己也是Shader初學者&#xff0c;因此可能會存在錯誤或者疏漏&#xff0c;如果您在Shader開發上有所心得&#xff0c;很歡迎并懇請您…

JVM:如何分析線程堆棧

英文原文&#xff1a;JVM: How to analyze Thread Dump 在這篇文章里我將教會你如何分析JVM的線程堆棧以及如何從堆棧信息中找出問題的根因。在我看來線程堆棧分析技術是Java EE產品支持工程師所必須掌握的一門技術。在線程堆棧中存儲的信息&#xff0c;通常遠超出你的想象&…

一個工科研究生畢業后的職業規劃

http://blog.csdn.net/wojiushiwo987/article/details/8592359一個工科研究生畢業后的職業規劃 [wojiushiwo987個人感觸]:說的很誠懇&#xff0c;對于馬上面臨畢業的我很受用&#xff0c;很有啟發。有了好的職業生涯規劃&#xff0c;才有了前進的方向和動力&#xff0c;才能…

SQLSERVER中如何忽略索引提示

SQLSERVER中如何忽略索引提示 原文:SQLSERVER中如何忽略索引提示SQLSERVER中如何忽略索引提示 當我們想讓某條查詢語句利用某個索引的時候&#xff0c;我們一般會在查詢語句里加索引提示&#xff0c;就像這樣 SELECT id,name from TB with (index(IX_xttrace_bal)) where bal…

JavaScript——以簡單的方式理解閉包

閉包&#xff0c;在一開始接觸JavaScript的時候就聽說過。首先明確一點&#xff0c;它理解起來確實不復雜&#xff0c;而且它也非常好用。那我們去理解閉包之前&#xff0c;要有什么基礎呢&#xff1f;我個人認為最重要的便是作用域&#xff08;lexical scope&#xff09;&…

jquery實現二級聯動不與數據庫交互

<select id"pro" name"pro" style"width:90px;"></select> <select id"city" name"city" style"width: 90px"></select> $._cityInfo [{"n":"北京市","c"…

[016]轉--C++拷貝構造函數詳解

一. 什么是拷貝構造函數 首先對于普通類型的對象來說&#xff0c;它們之間的復制是很簡單的&#xff0c;例如&#xff1a; [c-sharp] view plaincopy int a 100; int b a; 而類對象與普通對象不同&#xff0c;類對象內部結構一般較為復雜&#xff0c;存在各種成員變量。下…

js中調用C標簽實現百度地圖

<script type"text/javascript"> //json數組 var jsonArray document.getElementById("restaurant").value; var map new BMap.Map("milkMap"); // 創建地圖實例 <c:forEach items"${restaurantlist}" var"…

jquery較驗組織機構編碼

//*************************組織機構碼較驗************************* function checkOrganizationCode() { var weight [3, 7, 9, 10, 5, 8, 4, 2]; var str 0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ; var reg /^([0-9A-Z]){8}-[0-9|X]{1}$/; var organizationcode $("…

自定義GrildView實現單選功能

首先看實現功能截圖&#xff0c;這是一個自定義Dialog,并且里面內容由GrildView 綁定數據源&#xff0c;實現類似單選功能。 首先自定義Dialog&#xff0c;綁定數據源 自定義Dialog彈出框大小方法 最主要實現的就是點擊顏色切換的功能&#xff0c;默認GrildView的每一項都是藍色…

Java數字字符串如何轉化為數字數組

eg&#xff1a; String numberString "0123456789"; 如何轉化為&#xff1a;int[] digitArry new int[]{0,1,2,3,4,5,6,7,8,9}; 解決辦法&#xff1a; char[] digitNumberArray numberString.toCharArray(); int[] digitArry new int[digitString.toCharArray().l…

『重構--改善既有代碼的設計』讀書筆記----序

作為C的程序員&#xff0c;我從大學就開始不間斷的看書&#xff0c;看到如今上班&#xff0c;也始終堅持每天多多少少閱讀技術文章&#xff0c;書看的很多&#xff0c;但很難有一本書&#xff0c;能讓我去反復的翻閱。但唯獨『重構--改善既有代碼的設計』這本書讓我重復看了不下…

微信公共平臺接口開發--Java實現

Java微信實現&#xff0c;采用SpringMVC 架構&#xff0c;采用SAXReader解析XML RequestMapping(value"/extend") public class WeixinController { RequestMapping(value"/weixin") public ModelAndView weixin(HttpServletRequest request,HttpServlet…

最大權閉合圖hdu3996

定義&#xff1a;最大權閉合圖&#xff1a;是有向圖的一個點集&#xff0c;且該點集的所有出邊都指向該集合。即閉合圖內任意點的集合也在改閉合圖內&#xff0c;給每個點分配一個點權值Pu&#xff0c;最大權閉合圖就是使閉合圖的點權之和最大。 最小割建邊方式&#xff1a;源點…

非監督學習的單層網絡分析

這篇博客對應的是Andrew.Ng的那篇文章&#xff1a;An Analysis o f Single-Layer Networks in Unsupervised Feature Learning&#xff0c;文章的主要目的是討論receptive field size&#xff0c;number of hidden nodes&#xff0c; step-stride以及whitening在對卷積網絡模型…

Spring MVC 驗證碼

頁面 <% page language"java" import"java.util.*" pageEncoding"UTF-8"%> <% String path request.getContextPath(); String basePath request.getScheme()"://"request.getServerName()":"request.getServerP…

數據結構實驗之鏈表四:有序鏈表的歸并

數據結構實驗之鏈表四&#xff1a;有序鏈表的歸并 Time Limit: 1000MS Memory limit: 65536K 題目描述 分別輸入兩個有序的整數序列&#xff08;分別包含M和N個數據&#xff09;&#xff0c;建立兩個有序的單鏈表&#xff0c;將這兩個有序單鏈表合并成為一個大的有序單鏈表&…

apk文件編譯到系統文件中的方法(及包含so庫的)

把第三方或自己開發的apk文件編譯到系統文件(system.img)中的方法&#xff1a; 1 (1)源碼編譯后&#xff0c;把apk拷貝到out\target\product\generic\system\app中。 (2) 執行命令make snod , 把添加的spk編到system.img 中 缺點&#xff1a;執行make clean 后&#xff0c;再…