文件操作(打开关闭文件、文件顺序以及随机读写)
文章目录
- 写在前面
- 1. 文件的打开与关闭
- 1.1 文件指针
- 1.2 文件的打开(fopen)与关闭(fclose)
- 1.2.1 fopen函数
- 1.2.2 fclose函数
- 2. 文件的顺序读写
- 2.1. fgetc 和 fputc函数
- 2.1.1 fputc函数
- 2.1.2 fgetc函数
- 2.2 fgets 和 fputs函数
- 2.2.1 fputs函数
- 2.2.2 fgets函数
- 2.3 fscanf和fprintf函数
- 2.3.1 fprintf函数
- 2.3.2 fscanf函数
- 2.4 fread和fwrite函数
- 2.4.1 fwrite函数
- 2.4.2 fread函数
- 3. 文件的随机读写
- 3.1 fseek函数
- 3.2 ftell函数
- 3.3 rewind函数
写在前面
文件允许将程序的数据持久地保存在磁盘上。当程序结束运行或计算机关闭后,数据仍然存在,可以在下一次运行程序时继续使用。因此学习文件操作使我们能够更好地掌握数据的存储、读取和管理。
1. 文件的打开与关闭
1.1 文件指针
文件指针是C语言中用于处理文件的一个关键概念。它是一个指向文件流的指针,允许我们在文件中进行读取和写入操作。文件指针通常用于标识和跟踪文件的当前位置,以便进行文件的读取和写入操作。
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是有系统声明的,取名FILE。
例如,VS2013编译环境提供的 stdio.h 头文件中有以下的文件类型申明:
struct _iobuf {char *_ptr;int _cnt;char *_base;int _flag;int _file;int _charbuf;int _bufsiz;char *_tmpfname;};
typedef struct _iobuf FILE;
- 不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。
- 每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。
- 一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。

总的来说,通过文件指针变量能够找到与它关联的文件。
1.2 文件的打开(fopen)与关闭(fclose)
1.2.1 fopen函数
在C语言中,如果我们要打开一个文件,可以使用标准库函数fopen来打开文件。
函数原型:
FILE * fopen ( const char * filename, const char * mode );
- filename:要打开的文件的路径和名称。
- mode:打开文件的模式,以字符串形式传递,可以包括以下选项:
| 文件使用方式 | 含义 | 如果指定文件不存在 |
|---|---|---|
| “r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 |
| “w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
| “a”(追加) | 向文本文件尾添加数据 | 建立一个新的文件 |
| “rb”(只读) | 为了输入数据,打开一个二进制文件 | 出错 |
| “wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新的文件 |
| “ab”(追加) | 向一个二进制文件尾添加数据 | 出错 |
| “r+”(读写) | 为了读和写,打开一个文本文件 | 出错 |
| “w+”(读写) | 为了读和写,建议一个新的文件 | 建立一个新的文件 |
| “a+”(读写) | 打开一个文件,在文件尾进行读写 | 建立一个新的文件 |
| “rb+”(读写) | 为了读和写打开一个二进制文件 | 出错 |
| “wb+”(读写) | 为了读和写,新建一个新的二进制文件 | 建立一个新的文件 |
| “ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 建立一个新的文件 |
- 返回值:fopen函数打开文件成功,会返回一个指向FILE类型结构的指针,打开文件失败则返回NULL,因此在使用fopen打开文件的时候要对其返回值进行检查。
1.2.2 fclose函数
在C语言中,如果我们要关闭一个已经打开的文件,可以使用标准库函数fclose来关闭文件。
函数原型:
int fclose ( FILE * stream );
- stream:指向已打开文件的指针。
- 函数返回值:fclose函数返回一个整数值,返回0表示关闭文件成功,非0表示关闭文件出错。关闭文件后,文件指针将不再有效。
以下代码,展示使用fopen打开文件以及使用fclose关闭文件。
#include <stdio.h>int main()
{FILE* pf = fopen("data.txt", "w"); // 打开文件以写入模式if (pf == NULL) {perror("fopen");return 1;}// 写入数据到文件//.....// 关闭文件fclose(pf);pf = NULL;return 0;
}
2. 文件的顺序读写
以下是一些C语言标准库中常用的文件输入和输出函数,主要用于文件的顺序读写操作,按照文件中数据的顺序逐个读取或写入数据。
| 函数名 | 功能 | 适用于 |
|---|---|---|
| fgetc | 字符输入函数 | 所有输入流 |
| fputc | 字符输出函数 | 所有输出流 |
| fgets | 文本行输入函数 | 所有输入流 |
| fputs | 文本行输出函数 | 所有输出流 |
| fscanf | 格式化输入函数 | 所有输入流 |
| fprintf | 格式化输出函数 | 所有输出流 |
| fread | 二进制输入 | 文件 |
| fwrite | 二进制输出 | 文件 |
下面我们来一一介绍这些函数:
2.1. fgetc 和 fputc函数
fgetc 和 fputc 是C语言标准库提供的文件输入输出函数,它们从文件中读取或者向文件中写入一个字符。
2.1.1 fputc函数
该函数用于向指定文件流写入一个字符并移动光标到下一个位置。
函数原型:
int putc ( int character, FILE * stream );
- 参数 character :是要写入的字符,通常是字符的ASCII码值。
- 返回值:如果成功写入字符,返回写入的字符,如果发生写入错误,则返回 EOF。
以下代码,展示了使用putc向文件中写入字符:
#include <stdio.h>int main()
{FILE* pf = fopen("data.txt", "w");if (pf == NULL){perror("fopen");return 1;}//写文件fputc('a', pf);fputc('b', pf);fputc('c', pf);fputc('d', pf);//关闭文件fclose(pf);pf = NULL;
}
运行结果:

2.1.2 fgetc函数
函数用于从指定文件流(通常是通过 fopen 打开的文件)中读取一个字符。返回光标当前指向的字符。然后,将光标移动到下一个位置。
函数原型:
int fgetc ( FILE * stream );
- 返回值:返回所读取的字符的ASCII码值,如果已经读取到文件末尾或者发生错误,则返回EOF。
以下代码,展示了使用fgetc从文件中读取字符:
#include <stdio.h>int main()
{FILE* pf = fopen("data.txt", "r");if (pf == NULL){perror("fopen");return 1;}//读文件int ch = fgetc(pf);printf("ch = %c\n");//关闭文件fclose(pf);pf = NULL;
}
运行结果:

2.2 fgets 和 fputs函数
2.2.1 fputs函数
该函数用于向指定文件流中写入一个字符串。
函数原型:
int fputs ( const char * str, FILE * stream );
- 参数 str :是要写入的字符串。
- 参数 stream :是文件指针,通常是通过 fopen 打开的文件。
- 返回值:如果成功写入字符串,返回非负数;如果发生错误,则返回 EOF。
以下代码,展示了使用puts向文件中写入一个字符串:
#include <stdio.h>int main()
{FILE* pf = fopen("data.txt", "w");if (pf == NULL){perror("fopen");return 1;}//写文件fputs("hello ", pf);fputs("word", pf);//关闭文件fclose(pf);pf = NULL;return 0;
}
运行结果:

2.2.2 fgets函数
该函数用于从指定文件流中读取一行文本,并将其存储到字符数组中。
函数原型:
char * fgets ( char * str, int num, FILE * stream );
- 参数 str:是用于存储读取文本的字符数组。
- 参数 num:是要读取的最大字符数,函数最多读取num-1个字符。
- 参数 stream:是文件指针,通常是通过 fopen 打开的文件。
- 返回值:如果成功读取一行文本,返回 str;如果已经读取到文件末尾或发生错误,则返回 NULL。
以下代码,展示了使用fgets从文件中读取一行:
int main()
{FILE* pf = fopen("data.txt", "r");if (pf == NULL){perror("fopen");return 1;}//读文件char str[20] = { 0 };fgets(str, 11, pf);printf("%s\n", str);//关闭文件fclose(pf);pf = NULL;return 0;
}
运行结果:

2.3 fscanf和fprintf函数
2.3.1 fprintf函数
该函数用于将格式化数据写入指定文件流,类似于 printf 函数用于将格式化数据写入标准输出流。
函数原型:
int fprintf ( FILE * stream, const char * format, ... );
对比printf:
int printf ( const char * format, ... );
与printf相比fprintf多了一个参数stream,该参数是文件指针。
以下代码,展示了向文件中写入格式化的数据:
#include <stdio.h>struct A
{char a;int b;float c;
};int main()
{struct A a;a.a = 'x';a.b = 10;a.c = 3.14f;FILE* pf = fopen("data.txt", "w");if (pf == NULL){perror("fopen");return 1;}//写文件fprintf(pf, "a = %c b = %d c = %f", a.a, a.b, a.c);//关闭文件fclose(pf);pf = NULL;return 0;
}
运行结果:

2.3.2 fscanf函数
该函数用于从指定文件流中读取格式化数据,类似于 scanf 函数用于从标准输入中读取格式化数据。
函数原型:
int fscanf ( FILE * stream, const char * format, ... );
对比scanf:
int scanf ( const char * format, ... );
与scanf相比fscanf多了一个参数stream,该参数是文件指针。
以下代码,展示了从文件中读取格式化的数据:
#include <stdio.h>struct A
{char a;int b;float c;
};int main()
{struct A s;FILE* pf = fopen("data.txt", "r");if (pf == NULL){perror("fopen");return 1;}//读文件fscanf(pf, "a = %c b = %d c = %f", &a.a, &a.b, &a.c);//关闭文fclose(pf);pf = NULL;return 0;
}
运行结果:

2.4 fread和fwrite函数
fread 和 fwrite 是用来读写文件中的二进制数据。
2.4.1 fwrite函数
该函数用于ptr指向的内存块中的二进制数据写入指定文件流 stream。
函数原型:
size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );
- 参数 ptr:指向要写入的元素数组的指针。
- 参数 size:要写入的每个元素的大小(以字节为单位)。
- 参数 count:是要写入的元素的个数。
- 参数 stream:是文件指针,通常是通过 fopen 打开的文件。
- 返回值:返回实际成功写入的元素的个数。
以下代码,展示了向文件中写二进制的数据:
#include <stdio.h>struct A
{char a;int b;float c;
};int main()
{struct A a = {'q', 10, 3.14f};FILE* pf = fopen("data.txt", "wb");if (pf == NULL){perror("fopen");return 1;}//写文件fwrite(&a, sizeof(a), 1, pf);//关闭文件fclose(pf);pf = NULL;return 0;
}
运行结果:

2.4.2 fread函数
从指定文件流 stream中读取二进制数据并将其存储到ptr指向的内存块中。
函数原型:
size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );
- 参数ptr:指向大小至少为 (size*count) 字节的内存块的指针。
- 参数 size:要读取的每个元素的大小
- 参数 count:是要读取的元素个数。
- 参数 stream:是文件指针,通常是通过 fopen 打开的文件。
- 返回值:返回实际成功读取的元素个数,可能小于 count。
以下代码,展示了从文件中读取二进制的数据:
#include <stdio.h>struct A
{char a;int b;float c;
};int main()
{struct A a;FILE* pf = fopen("data.txt", "rb");if (pf == NULL){perror("fopen");return 1;}//读文件fread(&a, sizeof(a), 1, pf);//关闭文件fclose(pf);pf = NULL;
}
运行结果:

3. 文件的随机读写
文件的随机读取是指能够以非连续的方式访问文件中的数据,而不必从文件的开始逐个读取每个数据。
3.1 fseek函数
根据文件指针的位置和偏移量来定位文件指针。
函数原型:
int fseek ( FILE * stream, long int offset, int origin );
- stream:文件指针,通常是通过 fopen 打开的文件。
- offset:相对于 origin 的偏移量,以字节为单位。
- origin:用作偏移参考的位置,可以是 SEEK_SET(文件开头)、SEEK_CUR(当前位置)、或 SEEK_END(文件末尾)。
例子:
FILE* pf = fopen("data.txt", "r");
if (pf)
{fseek(pf, 5, SEEK_SET); // 将文件位置指针设置到文件开头后 5 字节的位置// 现在可以从这个位置读取数据fclose(pf);pf = NULL;
}
3.2 ftell函数
返回文件指针相对于起始位置的偏移量。
函数原型:
long int ftell ( FILE * stream );
- 返回值是当前位置相对于文件开头的偏移量。
例子:
#include <stdio.h>
int main()
{FILE* pf = fopen("data.txt", "r");if (pf){fseek(pf, 5, SEEK_SET);long pos = ftell(pf); // 获取当前位置的偏移量printf("当前位置: %ld 字节\n", pos);fclose(pf);pf = NULL;}return 0;
}
3.3 rewind函数
让文件指针的位置回到文件的起始位置。
函数原型:
void rewind ( FILE * stream );
例子:
#include <stdio.h>int main()
{FILE* pf = fopen("data.txt", "r");if (pf) {fseek(pf, 5, SEEK_SET);rewind(pf); // 重新设置文件位置指针到文件开头// 现在可以从文件开头读取数据fclose(pf);pf = NULL;}return 0;
}
至此,本片文章就结束了,若本篇内容对您有所帮助,请三连点赞,关注,收藏支持下。
创作不易,白嫖不好,各位的支持和认可,就是我创作的最大动力,我们下篇文章见!
如果本篇博客有任何错误,请批评指教,不胜感激 !!!

相关文章:
文件操作(打开关闭文件、文件顺序以及随机读写)
文章目录 写在前面1. 文件的打开与关闭1.1 文件指针1.2 文件的打开(fopen)与关闭(fclose)1.2.1 fopen函数1.2.2 fclose函数 2. 文件的顺序读写2.1. fgetc 和 fputc函数2.1.1 fputc函数2.1.2 fgetc函数 2.2 fgets 和 fputs函数2.2.1 fputs函数2.2.2 fgets函数 2.3 fscanf和fprin…...
HTTP 响应头 X-Frame-Options
简介 X-Frame-Options HTTP 响应头用来给浏览器一个指示。该指示的作用为:是否允许页面在 <frame>, </iframe> 或者 <object> 中展现。 网站可以使用此功能,来确保自己网站的内容没有被嵌套到别人的网站中去,也从而避免了…...
MongoDB 集群配置
一、副本集 Replica Sets 1.1 简介 MongoDB 中的副本集(Replica Set)是一组维护相同数据集的 mongod 服务。 副本集可提供冗余和高可用性,是所有生产部署的基础。 也可以说,副本集类似于有自动故障恢复功能的主从集群。通俗的讲就…...
random生成随机数的灵活运用
random返回的 [0,1) 之间的一个随即小数 思考:请写出获取 a-b 之间的一个随机整数,a,b均为整数,比如 a2 , b7 即返回一个数 x > [2,7]Math.random()*(b-a) 返回的就是 [0,b-a](int)(aMath.random()*(b-a1)) 》 (int)(2Math.random()*6) Ma…...
宏定义实现二进制数的奇偶位交换
思路分析 通过宏定义来实现二进制数的奇偶位交换,如果一个个遍历交换的话,那得算到猴年马月,这是我在网上看到的一个思路: 我们将每一位(整数在计算机里存储是4字节,32位)二进制数的奇数位保留…...
【ELK 使用指南】ELK + Filebeat 分布式日志管理平台部署
ELK和EFLK 一、前言1.1 日志分析的作用1.2 需要收集的日志1.3 完整日志系统的基本特征 二、ELK概述2.1 ELK简介2.2 为什么要用ELK?2.3 ELK的组件 三、ELK组件详解3.1 Logstash3.1.1 简介3.1.2 Logstash命令常用选项3.1.3 Logstash 的输入和输出流3.1.4 Logstash配置文件 3.2 E…...
传输层 | UDP协议、TCP协议
之前讲过的http与https都是应用层协议,当应用层协议将报文构建好之后就要将报文往下层传输层进行传递,而传输层就是负责将数据能够从发送端传到接收端。 再谈端口号 端口号(port)标识了一个主机上进行通信的不同的应用程序,在TCP/IP协议中&…...
Webmin(CVE-2019-15107)远程命令执行漏洞复现
漏洞编号 CVE-2019-15107 webmin介绍 什么是webmin Webmin是目前功能最强大的基于Web的Unix系统管理工具。管理员通过浏览器访问Webmin的各种管理功能并完成相应的管理动作http://www.webmin.com/Webmin 是一个用 Perl 编写的基于浏览器的管理应用程序。是一个基于Web的界面…...
嵌入式实时操作系统的设计与开发 (前后台系统)
前后台结构 前后台系统也称为中断驱动系统,其软件结构的显著特点是运行的程序有前台和后台之分。 在后台,一组程序按照轮询方式访问CPU;在前台,当用户的请求到达时,首先向CPU触发中断,然后将该请求转交给后…...
Macos数字音乐库:Elsten Software Bliss for Mac
Elsten Software Bliss for Mac是一款优秀的音乐管理软件,它可以帮助用户自动化整理和标记数字音乐库,同时可以自动识别音乐信息并添加标签和元数据。 此外,Bliss还可以修复音乐库中的问题,例如重复的音乐文件和缺失的专辑封面等…...
基于SpringBoot的校园周边美食探索及分享平台的设计与实现
文章目录 项目介绍主要功能截图:登录注册个人信息管理后台首页轮播图管理美食鉴赏我的好友管理我的收藏管理用户管理部分代码展示设计总结项目获取方式🍅 作者主页:超级无敌暴龙战士塔塔开 🍅 简介:Java领域优质创作者🏆、 简历模板、学习资料、面试题库【关注我,都给…...
GPT-4V的图片识别和分析能力
GPT-4V是OpenAI开发的大型语言模型,是GPT-4的升级版本。GPT-4V在以下几个方面进行了改进: 模型规模更大:GPT-4V的参数量达到了1.37T,是GPT-4的10倍。训练数据更丰富:GPT-4V的训练数据包括了1.56T的文本和代码数据。算…...
蓝桥杯(等差素数列,C++)
思路: 1、因为找的是长度为10,且公差最小的等差素数列,直接用枚举即可。 2、枚举用三重循环,第一重枚举首项,第二重枚举公差,第三重因为首项算一个,所以枚举九个等差素数。 代码:…...
Ceph 中的写入放大
新钛云服已累计为您分享769篇技术干货 介绍 Ceph 是一个开源的分布式存储系统,设计初衷是提供较好的性能、可靠性和可扩展性。 Ceph 独一无二地在一个统一的系统中同时提供了对象、块、和文件存储功能。 Ceph 消除了对系统单一中心节点的依赖,实现了无中…...
Mabatis-puls强于Mybatis的地方
Mabatis-puls与Mybatis都是优秀的Java持久化框架,但是Mabatis-puls相较于Mybatis有以下几个方面的优势: 性能更优:Mabatis-puls采用了Javassist技术,使得它在运行时比Mybatis更快速,尤其是在执行大量SQL的情况下&#…...
vue项目npm intall时发生版本冲突的解决办法
在日常使用命令npm install / npm install XX下载依赖的操作中,我经常会遇到无法解析依赖树的问题(依赖冲突) 当遇到这种情况的时候,可以通过以下命令完成依赖安装: npm install --legacy-peer-deps npm install xxx…...
tomcat多实例部署jenkins
tomcat多实例部署jenkins 文章目录 tomcat多实例部署jenkins1.简介:2.优缺点:3.工作原理:4.工作流程:5.tomcat多实例部署jenkins流程5.1.环境说明5.2.部署前准备工作5.3.多实例部署tomcat5.4.部署jenkins5.5.创建一个jenkins项目5…...
强连通分量+缩点
[图论与代数结构 701] 强连通分量 题目描述 给定一张 n n n 个点 m m m 条边的有向图,求出其所有的强连通分量。 注意,本题可能存在重边和自环。 输入格式 第一行两个正整数 n n n , m m m ,表示图的点数和边数。 接下来…...
如何做系统架构设计
文章目录 1、如何进行架构设计体系架构需求体系架构设计体系架构文档化体系架构复审体系架构实现体系架构演化 2、架构设计注意事项分治原则服务自治拥抱变化可维护性考虑依赖和限制阅读代码注意事项 3、最后 系统架构应该如何设计,从自己做架构的经历来分享一些体…...
L14D6内核模块编译方法
一、内核模块基础代码解析 一个内核模块代码错误仍然会导致的内核崩溃。 GPL协议:开源规定,使用内核一些函数需要 1、单内核的缺点 单内核扩展性差的缺点减小内核镜像文件体积,一定程度上节省内存资源提高开发效率不能彻底解决稳定性低的缺…...
Python|GIF 解析与构建(5):手搓截屏和帧率控制
目录 Python|GIF 解析与构建(5):手搓截屏和帧率控制 一、引言 二、技术实现:手搓截屏模块 2.1 核心原理 2.2 代码解析:ScreenshotData类 2.2.1 截图函数:capture_screen 三、技术实现&…...
visual studio 2022更改主题为深色
visual studio 2022更改主题为深色 点击visual studio 上方的 工具-> 选项 在选项窗口中,选择 环境 -> 常规 ,将其中的颜色主题改成深色 点击确定,更改完成...
镜像里切换为普通用户
如果你登录远程虚拟机默认就是 root 用户,但你不希望用 root 权限运行 ns-3(这是对的,ns3 工具会拒绝 root),你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案:创建非 roo…...
三体问题详解
从物理学角度,三体问题之所以不稳定,是因为三个天体在万有引力作用下相互作用,形成一个非线性耦合系统。我们可以从牛顿经典力学出发,列出具体的运动方程,并说明为何这个系统本质上是混沌的,无法得到一般解…...
智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
当仓库学会“思考”,物流的终极形态正在诞生 想象这样的场景: 凌晨3点,某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径;AI视觉系统在0.1秒内扫描包裹信息;数字孪生平台正模拟次日峰值流量压力…...
selenium学习实战【Python爬虫】
selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...
【Oracle】分区表
个人主页:Guiat 归属专栏:Oracle 文章目录 1. 分区表基础概述1.1 分区表的概念与优势1.2 分区类型概览1.3 分区表的工作原理 2. 范围分区 (RANGE Partitioning)2.1 基础范围分区2.1.1 按日期范围分区2.1.2 按数值范围分区 2.2 间隔分区 (INTERVAL Partit…...
AI,如何重构理解、匹配与决策?
AI 时代,我们如何理解消费? 作者|王彬 封面|Unplash 人们通过信息理解世界。 曾几何时,PC 与移动互联网重塑了人们的购物路径:信息变得唾手可得,商品决策变得高度依赖内容。 但 AI 时代的来…...
蓝桥杯 冶炼金属
原题目链接 🔧 冶炼金属转换率推测题解 📜 原题描述 小蓝有一个神奇的炉子用于将普通金属 O O O 冶炼成为一种特殊金属 X X X。这个炉子有一个属性叫转换率 V V V,是一个正整数,表示每 V V V 个普通金属 O O O 可以冶炼出 …...
快速排序算法改进:随机快排-荷兰国旗划分详解
随机快速排序-荷兰国旗划分算法详解 一、基础知识回顾1.1 快速排序简介1.2 荷兰国旗问题 二、随机快排 - 荷兰国旗划分原理2.1 随机化枢轴选择2.2 荷兰国旗划分过程2.3 结合随机快排与荷兰国旗划分 三、代码实现3.1 Python实现3.2 Java实现3.3 C实现 四、性能分析4.1 时间复杂度…...
