- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我最近在很多帖子中都看到有人试图读取这样的文件:
#include <stdio.h>
#include <stdlib.h>
int
main(int argc, char **argv)
{
char *path = "stdin";
FILE *fp = argc > 1 ? fopen(path=argv[1], "r") : stdin;
if( fp == NULL ) {
perror(path);
return EXIT_FAILURE;
}
while( !feof(fp) ) { /* THIS IS WRONG */
/* Read and process data from file… */
}
if( fclose(fp) != 0 ) {
perror(path);
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
最佳答案
我想提供一个抽象的高层视角。
并发性和同时性
I / O操作与环境交互。环境不是您的程序的一部分,也不在您的控制之下。该环境确实与您的程序“同时存在”。与所有并发事件一样,关于“当前状态”的问题也没有道理:在并发事件之间没有“同时性”的概念。状态的许多属性根本不会同时存在。
让我更精确地说:假设您想问“您是否有更多数据”。您可以询问并发容器或I / O系统。但是答案通常是不可行的,因此毫无意义。因此,如果容器说“是”,该怎么办?到您尝试读取时,它可能不再有数据。同样,如果答案为“否”,那么在您尝试阅读时,数据可能已经到达。结论是,根本没有像“我有数据”这样的属性,因为您无法对任何可能的答案做出有意义的举动。 (使用缓冲输入的情况要好一些,可以想象得到“是的,我有数据”可以构成某种保证,但是您仍然必须能够处理相反的情况。然后输出情况肯定和我描述的一样糟糕:您永远不知道该磁盘或网络缓冲区是否已满。)
因此,我们得出结论,询问一个I / O系统是否能够执行I / O操作是不可能的,实际上是不合理的。我们与之交互的唯一可能方式(就像与并发容器一样)是尝试操作并检查其成功还是失败。在与环境进行交互的那一刻,只有那时,您才能知道该交互是否确实可行,并且在这一点上,您必须致力于执行交互。 (如果需要,这是一个“同步点”。)
紧急行动
现在我们到EOF。 EOF是您从尝试的I / O操作获得的响应。这意味着您正在尝试读取或写入某些内容,但是这样做时您无法读取或写入任何数据,而是遇到了输入或输出的末尾。基本上对于所有I / O API都是如此,无论是C标准库,C ++ iostream还是其他库。只要I / O操作成功,您就根本无法知道将来的操作是否还会成功。您必须始终首先尝试该操作,然后响应成功或失败。
例子
在每个示例中,请仔细注意,我们首先尝试I / O操作,然后在有效时使用结果。还要注意,尽管每个示例中的结果采用不同的形状和形式,但我们始终必须使用I / O操作的结果。
C stdio,从文件中读取:
for (;;) {
size_t n = fread(buf, 1, bufsize, infile);
consume(buf, n);
if (n < bufsize) { break; }
}
n
,即已读取的元素数(可能少至零)。
scanf
:
for (int a, b, c; scanf("%d %d %d", &a, &b, &c) == 3; ) {
consume(a, b, c);
}
scanf
的返回值,即转换后的元素数。
for (int n; std::cin >> n; ) {
consume(n);
}
std::cin
本身,可以在布尔上下文中对其进行评估,并告诉我们流是否仍处于
good()
状态。
for (std::string line; std::getline(std::cin, line); ) {
consume(line);
}
std::cin
。
write(2)
刷新缓冲区:
char const * p = buf;
ssize_t n = bufsize;
for (ssize_t k = bufsize; (k = write(fd, p, n)) > 0; p += k, n -= k) {}
if (n != 0) { /* error, failed to write complete buffer */ }
k
,即写入的字节数。这里的要点是,我们只能知道在写操作之后写了多少字节。
getline()
char *buffer = NULL;
size_t bufsiz = 0;
ssize_t nbytes;
while ((nbytes = getline(&buffer, &bufsiz, fp)) != -1)
{
/* Use nbytes of data in buffer */
}
free(buffer);
nbytes
,直到并包括换行符的字节数(如果文件未以换行符结尾,则为EOF)。
-1
(而不是EOF!)。
std::string input = " 123 "; // example
std::istringstream iss(input);
int value;
if (iss >> value >> std::ws && iss.get() == EOF) {
consume(value);
} else {
// error, "input" is not parsable as an integer
}
iss
(流对象本身),以检查是否成功格式化了对
value
的提取。但是,在又消耗了空格之后,我们执行了另一个I / O /操作
iss.get()
,并期望它作为EOF失败,如果整个字符串已经被格式化提取所消耗,情况就是如此。
strto*l
函数相似的功能。
while(!eof)
是错误的,因为它测试不相关的内容并且无法测试您需要知道的内容。结果是您错误地执行了假定代码正在访问已成功读取的数据的代码,而实际上却从未发生过。
关于c - 为什么“while(!feof(file))”总是错误的?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58109728/
这是一个更大的电影 Sprite 程序的一部分,由于某种原因,程序在到达这个 while 循环时崩溃了,我不明白我的问题是什么。 #include #include #include #incl
这个问题已经有答案了: feof detecting false end of file (2 个回答) 已关闭10 年前。 我有一个函数应该读取文件并检测文件何时结束。 该函数当前仅读取倒数第二行并
我之前问过一个不同的问题,但我对这个问题的看法有点偏题,所以我提出了一个新问题,因为我问的是一个完全不同的问题。 我有一个函数可以读取文本文件中的给定行(由 ac 变量给出)。它执行行的读取,然后检查
大家好,我正在编写一个 c 程序,我试图一次从一个文本文件中打印两行,问题是当在最后一行时,如果行是奇数,3 行 5 行。它会打印最后一行两次。我找不到针对这个特定问题的 if 函数的比较。是 boo
这个问题在这里已经有了答案: Why is “while( !feof(file) )” always wrong? (5 个答案) 关闭 5 年前。 我是 C 的新手。我偶然发现了 feof 的某
我正在尝试通过套接字发送文件,方法是一次读取一个字符,直到缓冲区(字符数组)已满,发送它,然后重复直到文件末尾。但出于某种原因,feof 在文件结束前变为真(我相信可能是在缓冲区数组已满时)。 int
我是 C++ 的初学者,正在努力更好地理解 feof()。我读过 feof() 标志仅在尝试读取文件末尾后才设置为 true 很多次初学者如果执行类似 同时(!feof(文件))。不过,我想了解的是,
所以我做了一件简单的事情,首先我通过 ssh2_exec 执行命令(在成功验证后),然后读取变量中的答案。下面是我的代码(没有验证) try { $stdout_stream = ss
This link讲述为什么 feof() 用作循环的退出指示器是一件坏事。 不安全==>在 while 中进行 feof() 检查,并在 while 内进行 fgets() 检查。 安全==>让fg
我的程序中出现错误(应该找到字符和整数文件中的所有整数),其中“feof”不起作用(while循环永远不会结束)或没有读取/找到整数,除非整个文件充满了整数。 我的代码... #include
void suffix_strip(FILE *fp) { FILE *fp1; char word[20],ch; int len; int i=0,j; f
我正在使用 feof 计算文件中的行数,如下所示。完成后,我需要逐行循环遍历同一个文件,并执行另一个取决于行数的操作。但是,while (!feof(f)) 不会在同一个文件流上运行第二次。有没有办法
我有以下枚举和结构: enum Destination { unknown = 0, hosok, parlament, var }; struct Client { char name[30];
我正在使用 fgets(); 从文本文件中读取行 我有一个循环:while(!feof(fp)) 问题是,即使在我到达并读取最后一行之后,程序仍继续进入循环,并且在调试中文件 ptr 仅显示点而不是数
我这里真的有问题。从文件中读取字符时,我似乎并没有真正找到退出循环的最佳方法。我知道每个教程都建议我不应该使用 while( !feof() ) 但他们除了在 while 中放置 fgets() 之外
FILE *test; student st; int i = 0; test = fopen("example.bin", "rb"); while (feof(test)) { fread
我在从文件中读取行时遇到问题。基本上,我的代码多读了一行,所有变量都为“0”,即 如果我的文件是: 0, 0, 3, 64, 0, 0, 0, 0 2, 0, 6, 64, 0, 0, 0, 0 4,
这个问题在这里已经有了答案: Why is “while( !feof(file) )” always wrong? (5 个答案) 关闭 5 年前。 所以我阅读了多篇关于为什么 feof 不能正常
最近几天我一直在处理一个奇怪的 PHP 问题,feof() 函数在文件结束之前返回 true。下面是我的代码的框架: $this->fh = fopen("bigfile.txt", "r");
我打开一个文件,在指针ptr 的地址找到流。我正在尝试查看文件是否为空白。使用以下内容 if (fgetc(ptr) != EOF) 按预期工作。当文件为空时,不执行该语句。当文件不为空时,语句不执行
我是一名优秀的程序员,十分优秀!