判断读入数据文件结尾【从文件读入数据】

2024-01-22 10:08

本文主要是介绍判断读入数据文件结尾【从文件读入数据】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

参考:http://blog.csdn.net/rebel_321/article/details/4927464

在使用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;}    return 0;
}

编译并运行,

1、如果test.txt不存在,程序会形成死循环,fin.eof()永远返回false,
2、如果test.txt为空,程序打印出一个x字符,
3、当test.txt中存在一字符串“abcd”且没有换行时,程序打印出“abcdd”,
4、当存在以上字符串并且有一新的空行时,程序打印出“abcd”加上一空行。

在这里,可能有一个误区,认为eof()返回true时是读到文件的最后一个字符,其实不然,eof()返回true时是读到文件结束符0xFF,而文件结束符是最后一个字符的下一个字符。

如下图所示:

因此,当读到最后一个字符时,程序会多读一次(编译器会让指针停留在最后一个字符那里,然后重复读取一次;因此当存在字符串且最后无换行的时候最后一个字符会读取两次并输出两次)

要解决以上的问题,只需要调整一下条件语句即可:
   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;}    
e");return 0;
}

上述代码在VS2012下编译运行,发现,当文件结尾没有空行时,结果正确;

当结尾有空行时,整个字符串将被重复输出一次

若将条件语句修改为:

      fin >> str         

则结果正确

综上所述,
1. 判断文件是否为空时使用peek函数,若peek返回EOF则文件为空;
2. 读取文件过程中,读取非char型时,使用peek判断文件尾将不再适用,循环判断条件应改用>>操作符进行读取,若读入char型缓冲区,peek函数会表现得很好。

另外,其实类似与第二个方法,当做ACM之类的训练题需要为了方便读入数据自己建立文件存储数据时,还有一个更直接的常用方法(虽然不是最优→_→),就是当需要读入数据的时候,直接将读入的数据作为判断条件。

例如,输入数据第一行为数据个数,第二行为数据,此时可直接将第一行作为循环读入次数

测试程序:
#include "stdafx.h"
#include <iostream>
#include <fstream>
#include <vector>
using namespace std;
int main()
{int n,i = 0;ifstream cin("test.txt");cin>>n;while( n != -1 ){cout<<n<<endl;vector<char> a(n);while(n--){cin>>a[i];i++;}//输出读入结果作测试for(i =0; i < a.size(); i++)cout<<a[i];cout<<endl<<"success"<<endl<<endl;//输出结果i = 0;cin>>n;}return 0;
}
在cpp根目录建立a.txt,内容如下(随意)
3
21a
5
1sd45
9
985sdf#%6



这篇关于判断读入数据文件结尾【从文件读入数据】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/632649

相关文章

Python在二进制文件中进行数据搜索的实战指南

《Python在二进制文件中进行数据搜索的实战指南》在二进制文件中搜索特定数据是编程中常见的任务,尤其在日志分析、程序调试和二进制数据处理中尤为重要,下面我们就来看看如何使用Python实现这一功能吧... 目录简介1. 二进制文件搜索概述2. python二进制模式文件读取(rb)2.1 二进制模式与文本

利用c++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

C#实现将XML数据自动化地写入Excel文件

《C#实现将XML数据自动化地写入Excel文件》在现代企业级应用中,数据处理与报表生成是核心环节,本文将深入探讨如何利用C#和一款优秀的库,将XML数据自动化地写入Excel文件,有需要的小伙伴可以... 目录理解XML数据结构与Excel的对应关系引入高效工具:使用Spire.XLS for .NETC

MySQL数据目录迁移的完整过程

《MySQL数据目录迁移的完整过程》文章详细介绍了将MySQL数据目录迁移到新硬盘的整个过程,包括新硬盘挂载、创建新的数据目录、迁移数据(推荐使用两遍rsync方案)、修改MySQL配置文件和重启验证... 目录1,新硬盘挂载(如果有的话)2,创建新的 mysql 数据目录3,迁移 MySQL 数据(推荐两

Python数据验证神器Pydantic库的使用和实践中的避坑指南

《Python数据验证神器Pydantic库的使用和实践中的避坑指南》Pydantic是一个用于数据验证和设置的库,可以显著简化API接口开发,文章通过一个实际案例,展示了Pydantic如何在生产环... 目录1️⃣ 崩溃时刻:当你的API接口又双叒崩了!2️⃣ 神兵天降:3行代码解决验证难题3️⃣ 深度

MySQL快速复制一张表的四种核心方法(包括表结构和数据)

《MySQL快速复制一张表的四种核心方法(包括表结构和数据)》本文详细介绍了四种复制MySQL表(结构+数据)的方法,并对每种方法进行了对比分析,适用于不同场景和数据量的复制需求,特别是针对超大表(1... 目录一、mysql 复制表(结构+数据)的 4 种核心方法(面试结构化回答)方法 1:CREATE

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

MySQL中的DELETE删除数据及注意事项

《MySQL中的DELETE删除数据及注意事项》MySQL的DELETE语句是数据库操作中不可或缺的一部分,通过合理使用索引、批量删除、避免全表删除、使用TRUNCATE、使用ORDERBY和LIMI... 目录1. 基本语法单表删除2. 高级用法使用子查询删除删除多表3. 性能优化策略使用索引批量删除避免

MySQL 数据库进阶之SQL 数据操作与子查询操作大全

《MySQL数据库进阶之SQL数据操作与子查询操作大全》本文详细介绍了SQL中的子查询、数据添加(INSERT)、数据修改(UPDATE)和数据删除(DELETE、TRUNCATE、DROP)操作... 目录一、子查询:嵌套在查询中的查询1.1 子查询的基本语法1.2 子查询的实战示例二、数据添加:INSE

Linux服务器数据盘移除并重新挂载的全过程

《Linux服务器数据盘移除并重新挂载的全过程》:本文主要介绍在Linux服务器上移除并重新挂载数据盘的整个过程,分为三大步:卸载文件系统、分离磁盘和重新挂载,每一步都有详细的步骤和注意事项,确保... 目录引言第一步:卸载文件系统第二步:分离磁盘第三步:重新挂载引言在 linux 服务器上移除并重新挂p