【C语言】文件操作(二)

在这里插入图片描述

前言
在文件操作(一)中我们了解了文件及文件指针,还有文件的打开和关闭。在这篇博客我们学习顺序读写文件的几个函数。


一、文件的顺序读写

1.1 顺序读写函数介绍

函数名功能适用于
fgetc字符输⼊函数所有输⼊流
fputc字符输出函数所有输出流
fgets⽂本⾏输⼊函数所有输⼊流
fputs⽂本⾏输出函数所有输出流
fscanf格式化输⼊函数所有输⼊流
fprintf格式化输出函数所有输出流
fread⼆进制输⼊⽂件
fwrite⼆进制输出⽂件

fgetc 字符输入函数

int fgetc(FILE *stream)
参数——stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了要在上面执行操作的流。

返回值——该函数以无符号 char 强制转换为 int 的形式返回读取的字符(整数代表从文件流中读取的下一个字符的字符编码值),如果到达文件末尾或发生读错误,则返回 EOF

实例:

#include <stdio.h>

int main() {
    // 打开文件以供读取
    FILE *file = fopen("example.txt", "r");//"r"不会自动创建文件,需要我们自己在创建

    if (file == NULL) {
        perror("无法打开文件");
        return 1;
    }

    int character;
    
    // 使用 fgetc 逐字符读取文件内容
    while ((character = fgetc(file)) != EOF) {
        // 打印读取的字符
        putchar(character);
    }

    // 关闭文件
    fclose(file);

    return 0;
}

创建文本文档:
在这里插入图片描述
运行结果:
在这里插入图片描述


fputc 字符输出函数

int fputc(int char, FILE *stream)
参数——char -- 这是要被写入的字符。该字符以其对应的 int 值进行传递。
      stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了要被写入字符的流。

返回值——如果没有发生错误,则返回被写入的字符。如果发生错误,则返回 EOF,并设置错误标识符。

实例:

#include <stdio.h>

int main() {
    // 打开文件以供写入
    FILE *file = fopen("output.txt", "w");//"w"如果没有output.txt文件,建⽴⼀个新的⽂件
    if (file == NULL) {
        perror("无法打开文件");
        return 1;
    }

    // 要写入文件的字符
    char character = 'A';

    // 使用 fputc 将字符写入文件
    if (fputc(character, file) == EOF) {
        perror("写入文件时出错");
        return 1;
    }

    // 关闭文件
    fclose(file);

    return 0;
}

运行结果:
在这里插入图片描述


fgets 文本行输入函数

char *fgets(char *str, int n, FILE *stream)
参数——str -- 这是指向一个字符数组的指针,该数组存储了要读取的字符串。
      n -- 这是要读取的最大字符数(包括最后的空字符)。通常是使用以 str 传递的数组长度。
      stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了要从中读取字符的流。

返回值——如果成功,该函数返回相同的 str 参数。如果到达文件末尾或者没有读取到任何字符,str 的内容保持不变,并返回一个空指针。如果发生错误,返回一个空指针。

实例:

#include <stdio.h>

int main() {
    // 打开文件以供读取
    FILE* file = fopen("example.txt", "r");
    if (file == NULL) {
        perror("无法打开文件");
        return 1;
    }

    // 字符数组用于存储读取的文本行
    char buffer[13];

    // 使用 fgets 从文件中读取一行文本
    while (fgets(buffer, sizeof(buffer), file) != NULL) {
        // 打印读取的文本行
        printf("%s", buffer);
    } 

    // 关闭文件
    fclose(file);

    return 0;
}

创建文本文档:
在这里插入图片描述
运行结果:
在这里插入图片描述


fputs 文本行输出函数

int fputs(const char *str, FILE *stream)
参数——str -- 这是一个数组,包含了要写入的以空字符终止的字符序列。
      stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了要被写入字符串的流。

返回值——该函数返回一个非负值,如果发生错误则返回 EOF

实例:

#include <stdio.h>

int main() {
    // 打开文件以供写入
    FILE *file = fopen("output.txt", "w");
    if (file == NULL) {
        perror("无法打开文件");
        return 1;
    }

    // 要写入文件的字符串
    const char *text = "Hello, World!";

    // 使用 fputs 将字符串写入文件
    if (fputs(text, file) == EOF) {
        perror("写入文件时出错");
        return 1;
    }

    // 关闭文件
    fclose(file);

    return 0;
}

运行结果:
在这里插入图片描述


fread —二进制输入 fwrite —二进制输入

//fread
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream)
参数——ptr -- 这是指向带有最小尺寸 size*nmemb 字节的内存块的指针。
      size -- 这是要读取的每个元素的大小,以字节为单位。
      nmemb -- 这是元素的个数,每个元素的大小为 size 字节。
      stream -- 这是指向 FILE 对象的指针,该 FILE 对象指定了一个输入流。

返回值——成功读取的元素总数会以 size_t 对象返回,size_t 对象是一个整型数据类型。如果总数与 nmemb 参数不同,则可能发生了一个错误或者到达了文件末尾。
//fwrite
size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream)
参数——ptr -- 这是指向带有最小尺寸 size*nmemb 字节的内存块的指针。
      size -- 这是要读取的每个元素的大小,以字节为单位。
      nmemb -- 这是元素的个数,每个元素的大小为 size 字节。
      stream -- 这是指向 FILE 对象的指针,该 FILE 对象指定了一个输入流。
返回值——如果成功,该函数返回一个 size_t 对象,表示元素的总数,该对象是一个整型数据类型。如果该数字与 nmemb 参数不同,则会显示一个错误

实例:

//首先使用fwrite写入二进制数据,然后用fread读出
#include <stdio.h>

int main() {
    // 创建一个包含二进制数据的数组
    int data_to_write[5] = { 1, 2, 3, 4, 5 };

    // 打开一个二进制文件以供写入
    FILE* file_write = fopen("binary_data.bin", "wb");

    if (file_write == NULL) {
        perror("无法打开文件");
        return 1;
    }

    // 使用 fwrite 写入二进制数据到文件
    size_t elements_written = fwrite(data_to_write, sizeof(int), 5, file_write);

    if (elements_written < 5) {
        perror("写入文件时出错");
        fclose(file_write);
        return 1;
    }

    // 关闭文件
    fclose(file_write);

    // 打开同一个二进制文件以供读取
    FILE* file_read = fopen("binary_data.bin", "rb");

    if (file_read == NULL) {
        perror("无法打开文件");
        return 1;
    }

    // 从文件中读取二进制数据
    int data_read[5];

    size_t elements_read = fread(data_read, sizeof(int), 5, file_read);

    if (elements_read < 5) {
        perror("读取文件时出错");
        fclose(file_read);
        return 1;
    }

    // 关闭文件
    fclose(file_read);

    // 打印读取的数据
    printf("读取的数据:");
    for (size_t i = 0; i < elements_read; i++) {
        printf("%d ", data_read[i]);
    }
    printf("\n");

    return 0;
}

运行结果:
在这里插入图片描述
二进制文件:
在这里插入图片描述


1.2 比较一些函数

输入函数功能函数声明
scanf从标准输入(通常是键盘)读取格式化输入*int scanf(const char format, …);
fscanf从文件流中读取格式化输入int fscanf(FILE *stream, const char *format, …);
sscanf从字符串中读取格式化输入int sscanf(const char *str, const char *format, …);
输入函数功能函数声明
printf将格式化的输出写入到标准输出(通常是终端屏幕)int printf(const char *format, …);
fprintf将格式化的输出写入到文件int fprintf(FILE *stream, const char *format, …);
sprintf将格式化的输出写入到字符串int sprintf(char *str, const char *format, …);

可以看到,fscanf 和 sscanf 与 fprintf 和 sprintf ,跟scanf 和 sprintf 的使用是相似的,唯一不同的是,它们输入和输出函数从不同的源(标准输入、文件、字符串)读取数据,或将数据输出到不同的目标(标准输出、文件、字符串)。
实例:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>

int main() {
    int num;
    char name[20];
    FILE* file;

    // 输入函数示例
    printf("请输入一个整数:");
    scanf("%d", &num); // 从标准输入读取整数,并放入num

    file = fopen("input.txt", "r");
    if (file != NULL) {
    
        // 从文件中读取一个字符串,并放入name(提前创建input.txt,并将"Hello,"保存在文件中)
        fscanf(file, "%s", name); 
        fclose(file);
    }

    char input_string[] = "John";
    // 从字符串中解析整数和字符串,
    //已经在input.txt中读取了字符串"Hello,"并放在了name中
    //需要在name加上6个偏移量处读取input_string字符串,防止字符串被覆盖
    sscanf(input_string, "%s", name + 6);

    // 输出函数示例
    printf("整数:%d\n", num); // 输出到标准输出,预期结果  整数:20
    file = fopen("output.txt", "w");
    if (file != NULL) {
        fprintf(file, "整数:%d\n", num); // 输出到文件output.txt,预期结果  整数:20
        fclose(file);
    }

    char output_string[50];
    sprintf(output_string, "Hello,%s!", name); // 输出到字符串,预期结果  Hello,Hello,John!
    printf("%s\n", output_string);

    return 0;
}

创建文件:
在这里插入图片描述
运行结果:

在这里插入图片描述


在上一篇博客【C语言】文件操作(一)中我们讲到标准流的时候,我们说:实际上 stdin stdout stderr 都是在C标准库中定义的全局变量,具有与 FILE* 类型相同的类型。这使得它们可以作为参数传递给标准库函数。

为了证明这一点,举例:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>

int main() {
    int num;

    // 从标准输入(键盘)读取数据
    printf("请输入一个整数:");
    fscanf(stdin,"%d", &num);

    // 将错误消息写入标准错误
    if (num < 0) {
        fprintf(stderr, "错误:输入的数字不能为负数\n");
        return 1;
    }

    // 输出结果到标准输出
    fprintf(stdout,"您输入的数字是:%d\n", num);

    return 0;
}

在这里插入图片描述

但在实际应用中我们并不需要这样做,因为程序在启动的时候,标准流是默认打开的,scanf 和 printf就可以满足我们在从键盘输⼊数据,向屏幕上输出数据。


在这里插入图片描述
如果你喜欢这篇文章,点赞👍+评论+关注⭐️哦!
欢迎大家提出疑问,以及不同的见解。

  • 16
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 15
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 15
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值