在 C 语言中使用 fscanf 逐行读取文件
Jinku Hu
2023年10月12日
本文将介绍几种在 C 语言中使用 fscanf
逐行读取文件的方法。
在 C 语言中使用 fscanf
函数逐行读取文件
fscanf
函数是 C 标准库格式化输入工具的一部分。为不同的输入源提供了多个函数,如从 stdin
读取的 scanf
,从字符串读取的 sscanf
,以及从 FILE
指针流读取的 fscanf
。后者可用于逐行读取常规文件并将其存储在缓冲区中。
fscanf
采取与 printf
指定器类似的格式化规范,所有的格式化规范在这个页面上都有详细的列出。在下面的例子中,我们使用 fopen
函数调用打开样本输入文件,并分配满文件大小的内存将读流存储到其中。"%[^\n] "
格式字符串被指定为读取文件流,直到遇到新的行字符。fscanf
当到达输入结束时返回 EOF
,因此我们用 while
循环迭代,逐行打印。
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
const char *filename = "input.txt";
int main(int argc, char *argv[]) {
FILE *in_file = fopen(filename, "r");
struct stat sb;
stat(filename, &sb);
char *file_contents = malloc(sb.st_size);
while (fscanf(in_file, "%[^\n] ", file_contents) != EOF) {
printf("> %s\n", file_contents);
}
fclose(in_file);
exit(EXIT_SUCCESS);
}
输出:
> Temporary string to be written to file
请注意,即使前面的代码很可能在文件系统中存在输入文件名的情况下成功运行,但多次函数调用还是会失败,程序异常终止。接下来的示例代码是修改后的版本,其中实现了错误检查例程。
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
const char *filename = "input.txt";
int main(int argc, char *argv[]) {
FILE *in_file = fopen(filename, "r");
if (!in_file) {
perror("fopen");
exit(EXIT_FAILURE);
}
struct stat sb;
if (stat(filename, &sb) == -1) {
perror("stat");
exit(EXIT_FAILURE);
}
char *file_contents = malloc(sb.st_size);
while (fscanf(in_file, "%[^\n] ", file_contents) != EOF) {
printf("> %s\n", file_contents);
}
fclose(in_file);
exit(EXIT_SUCCESS);
}
输出:
> Temporary string to be written to file
使用 fscanf
函数用 C 语言逐字读取文件
利用 fscanf
函数的另一个有用的例子是遍历文件并解析每一个空格分隔的标记。请注意,与前面的例子相比,唯一需要改变的是格式指定符为"%[^\n ] "
。stat
系统调用是为了检索文件大小,该值被用来作为 malloc
参数传递,以分配缓冲区。这种方法对于某些场景来说可能会很浪费,但它能保证即使是最大的单行文件也能存储在缓冲区中。
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
const char *filename = "input.txt";
int main(int argc, char *argv[]) {
FILE *in_file = fopen(filename, "r");
if (!in_file) {
perror("fopen");
exit(EXIT_FAILURE);
}
struct stat sb;
if (stat(filename, &sb) == -1) {
perror("stat");
exit(EXIT_FAILURE);
}
char *file_contents = malloc(sb.st_size);
while (fscanf(in_file, "%[^\n ] ", file_contents) != EOF) {
printf("> %s\n", file_contents);
}
fclose(in_file);
exit(EXIT_SUCCESS);
}
输出:
> Temporary
> string
> to
> be
> written
> to
> file
作者: Jinku Hu