椋鸟C语言笔记#33:文件的顺序读写

2024-01-13 23:04

本文主要是介绍椋鸟C语言笔记#33:文件的顺序读写,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

萌新的学习笔记,写错了恳请斧正。


目录

光标(文件位置指示器)

文件的顺序读写

fgetc

使用实例

fputc

使用实例

fgets

fputs

使用实例

fscanf

fprintf

fread

fwrite

使用实例


光标(文件位置指示器)

我们平常打字的时候就知道光标这个概念,而程序在打开文件后也是有一个光标,用于控制文件的读写。我们一般说文件指针指向文件的哪个位置,其实就是光标(文件位置指示器)指向文件的某个位置,而这个信息也是保存在对应流的结构体中的。

读和写操作都是从光标的位置进行,而光标的位置也随着读写移动

文件的顺序读写

文件的顺序读写就是从光标的位置开始,按顺序向后读写内容

文件顺序读写时我们没法随意控制光标的位置,只能让系统自己控制光标

fgetc
#include <stdio.h>
int fgetc( FILE* stream );

fgetc是字符输入函数,从指定的输入流读取下一个字符,并返回这个字符

如果读取失败则返回EOF,但是遇到文件尾导致的读取失败会设置流结构体上的文件尾指示器(这是有用的,后面feof函数会讲),而其他错误导致的失败会设置流结构体上的错误指示器(这在ferror函数中用得到)

其实getchar其实就可以理解为fgetc(stdin),从标准流读一个字符

但是注意:尽量使stream不要是一个带有副作用的表达式(它被重复计算)

使用实例

其中的错误处理部分暂且不表,后面讲

#include <stdio.h>
#include <stdlib.h>int main()
{FILE* fp = fopen("test.txt", "r");if (!fp){perror("File opening failed");return EXIT_FAILURE;}int c;  // 注意:int,非char,要求处理EOFwhile ((c = fgetc(fp)) != EOF)  // 标准C I/O读取文件循环{ putchar(c);}if (ferror(fp))puts("I/O error when reading");else if (feof(fp))puts("End of file reached successfully");fclose(fp);fp = NULL;return EXIT_SUCCESS;
}
fputc
#include <stdio.h>
int fputc( int ch, FILE* stream );

fputc是字符输出函数,向指定的输入流输出字符,并返回这个字符

如果输出失败则返回EOF并设置流结构体上的错误指示器

其实putchar(ch)其实就可以理解为fputc(ch, stdin),向标准流输出一个字符

但是注意:尽量使stream不要是一个带有副作用的表达式(它被重复计算)

使用实例

其中的错误处理部分暂且不表,后面讲

#include <stdio.h>
#include <stdlib.h>int main()
{int ret_code = 0;for (char c = 'a'; (ret_code != EOF) && (c <= 'z'); c++)ret_code = putc(c, stdout);/* 测试是否抵达 EOF */if (ret_code == EOF)if (ferror(stdout)){perror("putc()");fprintf(stderr, "putc() failed in file %s at line # %d\n", __FILE__, __LINE__ - 7);exit(EXIT_FAILURE);}putc('\n', stdout);return EXIT_SUCCESS;
}
fgets
#include <stdio.h>
char* fgets( char* str, int count, FILE* stream );

fgets是文本行输入函数,从指定的输入流读取最多count-1个字符并将其存储在str所指向的字符数组(以覆写的形式,不会清除原本内容),若是遇到文件尾或换行符则提前结束输入。如果是遇到换行符结束,那么str的结尾也将包含一个换行符;如果是遇到文件尾结束,那么str的结尾没有换行符。只要没有出错,读取的字符串结尾都会补上'\0'。

  • 如果读取成功,则返回值就是str
  • 如果一个字符都没读取就直接遇到文件尾,则判定为失败,返回NULL且不会以'\0'覆写str的首字节,同时设置流结构体的文件尾指示器
  • 如果遇到其他错误导致失败,则设置流结构体的错误指示器和errno,此时str数组会发生什么谁也不知道

count小于等于1的情况是未定义的,不要这么做

fputs
#include <stdio.h>
int fputs( const char* str, FILE* stream );

fputs是文本行输出函数,向指定的输出流(stream)输出以'\0'结尾的字符串str的每一个字符(空字符不被写入)。如果输出成功,返回一个非负值(未定义,有些编译器返回写入的字符数,有些返回最后一个字符);如果输出失败,返回EOF并设置流结构体的错误指示器和errno。

fputs与puts还有一个区别,puts在输出后会再多输出一个换行符,fputs不会

使用实例
#include <stdio.h>int main()
{int rc = fputs("Hello world!", stdout);if (rc == EOF)perror("fputs()");return EXIT_SUCCESS;
}
fscanf
#include <stdio.h>
int fscanf( FILE* stream, const char* format, ... );

fscanf相比scanf除了可以选择从任意流stream输入外就没有什么区别了,都有被读字符串(应当以'\0'结尾)和可变参数列表。返回值为成功接受参数的数量(可以为0)。

直接遇到文件尾或读取失败则返回EOF,并设置流结构体上相关指示器

fprintf
#include <stdio.h>
fprintf( FILE* stream, const char* format, ... );

fprintf相比printf除了可以选择向任意流stream输出外就没有什么区别了,都有格式控制字符串和可变参数列表。返回值为输出到流的字符数,如果输出错误或者编码错误则返回一个负值(未定义)。

fread
#include <stdio.h>
size_t fread( void* buffer, size_t size, size_t count, FILE* stream );

fread是二进制输入函数,从指定流stream读取最多count个大小为size字节的对象到数组buffer中。读取成功则返回读取的对象数;读取出现错误或文件尾,返回值是小于count的未定义值并设置流结构体上相关指示器;如果size或count是零则返回零且不进行其他动作。

官网上描述如下:

如同转译每个对象为 unsigned char 数组,并对每个对象调用 size 次 fputc 以将那些 unsigned char 按顺序写入 stream 一般写入。文件位置指示器前进写入的字节数。

fwrite

fread是二进制输出函数,从数组buffer向指定流stream输出最多count个大小为size字节的对象。输出成功则返回读取的对象数;输出出现错误,返回值是小于count的未定义值如果size或count是零则返回零且不进行其他动作。

官网上描述如下:

如同转译每个对象为 unsigned char 数组,并对每个对象调用 size 次 fputc 以将那些 unsigned char 按顺序写入 stream 一般写入。文件位置指示器前进写入的字节数。

使用实例

 其中fseek函数用于移动光标,后面随机读写会讲

#include <stdio.h>
#include <stdlib.h>struct Stu
{char name[20];int age;float score;
};int main()
{struct Stu s = { "zhangsan", 20, 99.9 };struct Stu d = { 0 };FILE* pf = fopen("dat.txt", "w+b");if (!pf){perror("fopen");return EXIT_FAILURE;}fwrite(&s, sizeof(s), 1, pf);fseek(pf, 0, SEEK_SET);fread(&d, sizeof(s), 1, pf);printf("%s %d %.1f", d.name, d.age, d.score);fclose(pf);pf = NULL;return EXIT_SUCCESS;
}

这篇关于椋鸟C语言笔记#33:文件的顺序读写的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/603097

相关文章

ShardingProxy读写分离之原理、配置与实践过程

《ShardingProxy读写分离之原理、配置与实践过程》ShardingProxy是ApacheShardingSphere的数据库中间件,通过三层架构实现读写分离,解决高并发场景下数据库性能瓶... 目录一、ShardingProxy技术定位与读写分离核心价值1.1 技术定位1.2 读写分离核心价值二

GO语言短变量声明的实现示例

《GO语言短变量声明的实现示例》在Go语言中,短变量声明是一种简洁的变量声明方式,使用:=运算符,可以自动推断变量类型,下面就来具体介绍一下如何使用,感兴趣的可以了解一下... 目录基本语法功能特点与var的区别适用场景注意事项基本语法variableName := value功能特点1、自动类型推

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

Go语言使用Gin处理路由参数和查询参数

《Go语言使用Gin处理路由参数和查询参数》在WebAPI开发中,处理路由参数(PathParameter)和查询参数(QueryParameter)是非常常见的需求,下面我们就来看看Go语言... 目录一、路由参数 vs 查询参数二、Gin 获取路由参数和查询参数三、示例代码四、运行与测试1. 测试编程路

Python学习笔记之getattr和hasattr用法示例详解

《Python学习笔记之getattr和hasattr用法示例详解》在Python中,hasattr()、getattr()和setattr()是一组内置函数,用于对对象的属性进行操作和查询,这篇文章... 目录1.getattr用法详解1.1 基本作用1.2 示例1.3 原理2.hasattr用法详解2.

Go语言使用net/http构建一个RESTful API的示例代码

《Go语言使用net/http构建一个RESTfulAPI的示例代码》Go的标准库net/http提供了构建Web服务所需的强大功能,虽然众多第三方框架(如Gin、Echo)已经封装了很多功能,但... 目录引言一、什么是 RESTful API?二、实战目标:用户信息管理 API三、代码实现1. 用户数据

Go语言网络故障诊断与调试技巧

《Go语言网络故障诊断与调试技巧》在分布式系统和微服务架构的浪潮中,网络编程成为系统性能和可靠性的核心支柱,从高并发的API服务到实时通信应用,网络的稳定性直接影响用户体验,本文面向熟悉Go基本语法和... 目录1. 引言2. Go 语言网络编程的优势与特色2.1 简洁高效的标准库2.2 强大的并发模型2.

Go语言使用sync.Mutex实现资源加锁

《Go语言使用sync.Mutex实现资源加锁》数据共享是一把双刃剑,Go语言为我们提供了sync.Mutex,一种最基础也是最常用的加锁方式,用于保证在任意时刻只有一个goroutine能访问共享... 目录一、什么是 Mutex二、为什么需要加锁三、实战案例:并发安全的计数器1. 未加锁示例(存在竞态)