如何解决fgets在数组中存储什么
所以我在Kernighan和Ritchie的书中寻找了fgets()。就像这样:
char *fgets(char *s,int n,FILE *stream)
fgets最多将下一个n-1个字符读入数组s,如果遇到换行符则停止;换行符包含在以'\ 0'结尾的数组中。 fgets返回s,如果出现文件结尾或错误,则返回NULL。
所以我的困惑是,如果文件包含以下几行
Earth is the third planet from the Sun
The only astronomical object known to harbor life
About 29% of Earth's surface is land consisting of continents and islands
如果使用fgets()读取了文件,那么's'包含什么?
会包含"Earth is the third planet from the Sun\n\0"
还是会包含"Earth is the third planet from the Sun\0\n"
或其他内容?
解决方法
这是C标准本身对fgets
的描述-
§7.21.7.2
fgets函数从n中指定的字符数中读取的字符数最多少一个。 流指向的流进入s指向的数组。后面没有读取其他字符 换行符(保留)或文件末尾。立即写空字符 最后一个字符读入数组后。
在这里回答您的问题要注意两点。
-
fgets函数从stream指向的流到s指向的数组中读取的字符数最多少于n指定的字符数。
因此,您传递的
n
很重要,它表示要读取的最大字符数,包括空终止符。因此,如果您将
100
作为n
进行传递,它将从文件中读取最多 个99个字符,并将'\0'
放在文件末尾那。“最多”是什么意思?好吧,
fgets
将在遇到新行时停止(或者显然如果文件刚刚结束)。在换行符(保留)或文件结尾之后,不会读取其他字符
这意味着
fgets
将从文件中读取直到并包括第一个换行符\n
或最多n - 1
个字符-以先发生的为准 -
在将最后一个字符读入数组之后,立即将空字符写入。
这意味着,在从文件中读取所有字符完成之后,根据前面提到的条款,
\0
被写入缓冲区。
如果要进行fgets(s,1024,file)
,请在示例文件中回答您的问题-假设s
是指向可以容纳至少的字符数组/缓冲区的指针1024个字符,file
是文件的FILE*
-您会得到结果
Earth is the third planet from the Sun\n\0
由于fgets
在第一个\n
处停止,因为在通过n
所指示的char限制之前发生了这种情况。然后将\0
放在缓冲区的末尾。
请注意,fgets
不仅会写入您传递给它的缓冲区,而且还会返回相同的缓冲区-不要为此感到困惑。它们是指向相同位置的指针。
好的,让我们写一些代码来查看存储在数组中的内容。
#include <stdio.h>
int main(void) {
char s[64];
int size = sizeof(s);
int i;
/* initialize array */
for (i = 0; i < size; i++) s[i] = (char)0xff;
/* call fgets() */
fgets(s,size,stdin);
/* print contents of the array */
for (i = 0; i < size; i++) {
printf("%02X ",(unsigned char)s[i]);
if ((i + 1) % 16 == 0) {
int j,offset = (i / 16) * 16;
for (j = 0; j < 16; j++) {
putchar(0x20 <= s[offset + j] && s[offset + j] < 0x7f ? s[offset + j] : '.');
}
putchar('\n');
}
}
return 0;
}
输入(在问题中指定):
Earth is the third planet from the Sun
The only astronomical object known to harbor life
About 29% of Earth's surface is land consisting of continents and islands
45 61 72 74 68 20 69 73 20 74 68 65 20 74 68 69 Earth is the thi
72 64 20 70 6C 61 6E 65 74 20 66 72 6F 6D 20 74 rd planet from t
68 65 20 53 75 6E 0A 00 FF FF FF FF FF FF FF FF he Sun..........
FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ................
它包含"Earth is the third planet from the Sun\n\0"
。
让我们检查一下指定的缓冲区长度小于存储所需大小的情况。
用fgets(s,stdin);
代替fgets(s,32,stdin);
,我得到了output:
45 61 72 74 68 20 69 73 20 74 68 65 20 74 68 69 Earth is the thi
72 64 20 70 6C 61 6E 65 74 20 66 72 6F 6D 20 00 rd planet from .
FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ................
FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ................
这表明在这种情况下,它将存储n-1
个字符和1个终止的空字符。
如您所说,char *fgets(char *str,int n,FILE *stream)
从指定的流(stdin,文件..)中读取一行,函数读取行并在达到文件结尾或停止时停止读取(n-1)个字符。
因此,在您的示例中,s
将包含Earth is the third planet from the Sun\n\0
另一种情况是行中的字符数大于(n-1)。令n = 10,则s
将包含Earth is \0
。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。