在 C 语言中使用 fscanf 逐行读取文件

Jinku Hu 2023年1月30日 2021年2月28日
  1. 在 C 语言中使用 fscanf 函数逐行读取文件
  2. 使用 fscanf 函数用 C 语言逐字读取文件
在 C 语言中使用 fscanf 逐行读取文件

本文将介绍几种在 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
Author: Jinku Hu
Jinku Hu avatar Jinku Hu avatar

Founder of DelftStack.com. Jinku has worked in the robotics and automotive industries for over 8 years. He sharpened his coding skills when he needed to do the automatic testing, data collection from remote servers and report creation from the endurance test. He is from an electrical/electronics engineering background but has expanded his interest to embedded electronics, embedded programming and front-/back-end programming.

LinkedIn

相关文章 - C File