文件操作(详解)

该片博客有点长大家可以通过目录选择性阅读

这是个人主页

敲上瘾-CSDN博客

目录

1. 为什么使⽤⽂件?

2. 什么是⽂件?

2.1 程序⽂件

2.2 数据⽂件

2.3 ⽂件名

3. ⼆进制⽂件和⽂本⽂件?

4. ⽂件的打开和关闭

4.1 流和标准流

4.1.1 流

4.1.2 标准流

4.2 ⽂件指针

4.3 ⽂件的打开和关闭

 5.文件的顺序读写

5.1 fputc函数

5.2 fputs函数

5.3 fgetc函数

5.4 fgets函数 

5.5 fprintf函数

5.6 fscanf函数

​编辑

5.7 fwrite

​编辑

5.8 fread

6.sprintf函数

7.sscanf函数

8.文件的随机读写:

8.1 fseek函数

8.2 ftell函数

8.3 rewind函数

9.错误检测 

9.1 feof函数

9.2 ferror函数

10.缓冲区

10.1文件缓冲区


1. 为什么使⽤⽂件?

如果没有⽂件,我们写的程序的数据是存储在电脑的内存中,如果程序退出内存回收数据就失
了,等再次运⾏程序,是看不到上次程序的数据的,如果要将数据进⾏持久化的保存,我们可以使⽤⽂件。

2. 什么是⽂件?

磁盘(硬盘)上的⽂件是⽂件。
在程序设计中的⽂件有两种:程序⽂件数据⽂件(从⽂件功能的⻆度来的)。

2.1 程序⽂件

程序⽂件包括源程序⽂件(后缀为.c),⽬标⽂件(windows环境后缀为.obj),可执⾏程序(windows
环境后缀为.exe)。

2.2 数据⽂件

⽂件的内容不⼀定是程序,⽽是程序运⾏时读写的数据,⽐如程序运⾏需要从中读取数据的⽂件,或者输出内容的⽂件。
有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使⽤,这⾥处
理的就是磁盘上⽂件。

2.3 ⽂件名

⼀个⽂件要有⼀个唯⼀的⽂件标识,以便⽤⼾识别和引⽤。
⽂件名包含3部分:⽂件路径+⽂件名主⼲+⽂件后缀
例如: c:\code\test.txt
为了⽅便起⻅,⽂件标识常被称为⽂件名。

3. ⼆进制⽂件和⽂本⽂件?

根据数据的组织形式,数据⽂件被称为⽂本⽂件或者⼆进制⽂件
数据在内存中以⼆进制的形式存储,如果不加转换的输出到外存的⽂件中,就是⼆进制⽂件。
如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的⽂件就是⽂本⽂件。
⼀个数据在⽂件中是怎么存储的呢?
字符⼀律以ASCII形式存储,数值型数据既可以⽤ASCII形式存储,也可以使⽤⼆进制形式存储。
如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占⽤5个字节(每个字符⼀个字节),⽽⼆进制形式输出,则在磁盘上只占4个字节(VS2019测试)。

4. ⽂件的打开和关闭

4.1 流和标准流

4.1.1 流

我们程序的数据需要输出到各种外部设备,也需要从外部设备获取数据,不同的外部设备的输⼊输出操作各不相同,为了⽅便程序员对各种设备进⾏⽅便的操作,我们抽象出了流的概念,我们可以把流想象成流淌着字符的河。这样我们只需要对流进行操作。
C程序针对⽂件、画⾯、键盘等的数据输⼊输出操作都是通过流操作的。
⼀般情况下,我们要想向流⾥写数据,或者从流中读取数据,都是要打开流,然后操作。

4.1.2 标准流

那为什么我们从键盘输⼊数据,向屏幕上输出数据,并没有打开流呢?
那是因为C语⾔程序在启动的时候,默认打开了3个流:
stdin-标准输⼊流,在⼤多数的环境中从键盘输⼊,scanf函数就是从标准输⼊流中读取数据。
stdout-标准输出流,⼤多数的环境中输出⾄显⽰器界⾯,printf函数就是将信息输出到标准输出
流中。
stderr-标准错误流,⼤多数环境中输出到显⽰器界⾯。
这是默认打开了这三个流,我们使⽤scanf、printf等函数就可以直接进⾏输⼊输出操作的。
stdin、stdout、stderr三个流的类型是: FILE* ,通常称为⽂件指针
C语⾔中,就是通过 FILE* 的⽂件指针来维护流的各种操作的。

4.2 ⽂件指针

缓冲⽂件系统中,关键的概念是“⽂件类型指针”,简称“⽂件指针”。
每个被使⽤的⽂件都在内存中开辟了⼀个相应的⽂件信息区,⽤来存放⽂件的相关信息(如⽂件的名字,⽂件状态及⽂件当前的位置等)。这些信息是保存在⼀个结构体变量中的。该结构体类型是由系统声明的,取名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结构的变量,这样使⽤起来更加⽅便。
下⾯我们可以创建⼀个FILE*的指针变量:
        FILE* pf;//⽂件指针变量
定义pf是⼀个指向FILE类型数据的指针变量。可以使pf指向某个⽂件的⽂件信息区(是⼀个结构体变
量)。通过该⽂件信息区中的信息就能够访问该⽂件。也就是说,通过⽂件指针变量能够间接找到与
它关联的⽂件

4.3 ⽂件的打开和关闭

⽂件在读写之前应该先打开⽂件,在使⽤结束之后应该关闭⽂件。
在编写程序的时候,在打开⽂件的同时,都会返回⼀个FILE*的指针变量指向该⽂件,也相当于建⽴了
指针和⽂件的关系。
ANSI C 规定使⽤ fopen 函数来打开⽂件, fclose 来关闭⽂件。

​
int main()
{FILE* pf = fopen("text.txt", "w");//"w"表示以读的方式打开fclose(pf);//关闭return 0;
}​

 5.文件的顺序读写

相关函数:

  

5.1 fputc函数

fputc的参数如下

它的功能是把字符character输出到stream指向的文件中,字符的本质就是它的ascll值所以这里用int类型接收。如果写入成功则返回写入的字符的ascll码,失败则返回-1(即EOF)

示例代码:

#include<stdio.h>
#include<errno.h>
int main()
{FILE* pf = fopen("test.txt", "w");if (pf == NULL){perror(fopen);return 1;}for (char i = 'a'; i <= 'z'; i++){fputc(i, pf);}fclose(pf);return 0;
}

注意:当不存在test.txt这个文件的时候,以写的形式打开(即"w")程序执行后会创建一个test.txt文件,但如果以读的形式打开(即"r")的话程序会报错 。

注意:如果test.txt这个文件存在的时候,以写的形式打开(即"w")程序执行后会把原有的数据清空进行写入。

注意:不能以读的形式打开然后去写,也不能以写的形式打开然后去读。

一下方式可以找到这个被读写文件

5.2 fputs函数

fputs参数如下:

fputs的功能是把字符串str输出到stream指向的文件中,如果写入成功返回一个非负整数,写入失败,则返回-1(即EOF)。

示例代码:

int main()
{FILE* pf = fopen("test.txt", "w");if (pf == NULL){perror(fopen);return 1;}fputs("1234567", pf);char s[] = "abcdef";fputs(s, pf);fclose(pf);return 0;
}

5.3 fgetc函数

fgetc的参数如下:

它的作用是返回stream对应的文件里面的一个字符(它的返回值),所以需要一个字符变量来接收,当读取失败时返回-1,虽然说是返回字符,但fgetc函数的返回类型是int,不过这没关系每个字符都有对应的ascll码值,它的ascll是int类型,这里返回类型用int接收也是为了对应当读取失败时返回的-1,每读取一次stream指向的文件内容,文件指针往后移动一位

示例代码

#include<stdio.h>
int main()
{FILE* pf = fopen("test.txt", "r");if (!pf){perror(fopen);return 1;}char c;while ((c = fgetc(pf))!=-1){printf("%c", c);}fclose(pf);return 0;
}

5.4 fgets函数 

fgets参数如下:

fgets函数的功能是把stream指向的文件中的num个字符(即一个字符串)输入到str中,返回值是该字符串首字符的地址,当读取失败时返回NULL,每读取一次stream指向的文件内容,stream往后移动num位。

#include<stdio.h>
int main()
{FILE* pf = fopen("test.txt", "r");if (pf == NULL){perror(fopen);return 1;}char s[30] = { 0 };while(fgets(s,2,pf)!=NULL){printf(s);}return 0;
}

5.5 fprintf函数

参数如下:

它的功能是把数据以格式化的形式输出到stream指向的文件

示例代码

#include<stdio.h>
#include<errno.h>
int main()
{FILE* pf = fopen("test.txt", "w");if (pf == NULL){perror(fopen);return 1;}int a = 6;char s[10] = "pupm";fprintf(pf, "%d %s", a, s);fclose(pf);return 0;
}

5.6 fscanf函数

参数如下:

它的作用是把stream指向的文件的内容格式化的输入到内存中。

示例代码

#include<stdio.h>
#include<errno.h>
int main()
{FILE* pf = fopen("test.txt", "r");if (pf == NULL){perror(fopen);return 1;}int a = 0;char s[10] = {0};fscanf(pf,"%d %s", &a, s);printf("%d %s", a,s);fclose(pf);return 0;
}

5.7 fwrite

fwrite的功能是把ptr指向的空间中的count个元素以二进制的形式写入stream所指向的文件中,其中参数size表示ptr指向的空间中的一个元素占的字节数,返回成功写入的元素个数

注意:文件要以"wb"的形式打开,表示以二进制形式写入。 

示例代码

int main()
{FILE* pf = fopen("test.txt", "wb");if (pf == NULL){perror(fopen);return 1;}int arr[10] = {1,2,3,4,5,6,7,8,9,10};fwrite(arr, sizeof(arr[0]), 8, pf);fclose(pf);return 0;
}

因为是以二进制的形式写进去的,所以里面的内容是看不懂的 

5.8 fread

参数如下:

与fwrite函数相反该函数的功能是把stream所指向的文件内容的count个元素以二进制的形式读取到ptr所指向的空间,其中参数size表示ptr指向的空间中的一个元素占的字节数,返回成功读取到的元素个数

 注意:文件要以"rb"的形式打开,表示以二进制形式读取。

示例代码

int main()
{FILE* pf = fopen("test.txt", "rb");if (pf == NULL){perror(fopen);return 1;}int arr[10] = { 0 };int i = 0;while (fread(&arr[i], sizeof(arr[0]), 1, pf) == 1){printf("%d ", arr[i++]);}fclose(pf);return 0;
}

6.sprintf函数

参数如下:

功能是把格式的数据转化为字符串str

示例代码

#include<stdio.h>
int main()
{char s[10] = { 0 };int a = 42;char c='a';sprintf(s, "%d %c", a, c);printf("%s", s);return 0;
}

7.sscanf函数

sscanf的功能是在字符串s中数据以格式化的形式读取。

示例代码

#include<stdio.h>
int main()
{char s[10] = "4k";int a;char c;sscanf(s, "%d%c", &a, &c);printf("%d %c", a, c);return 0;
}

8.文件的随机读写:

相关函数

fseek     根据⽂件指针的位置和偏移量来定位⽂件指针

ftell        返回⽂件指针相对于起始位置的偏移量

rewind   让⽂件指针的位置回到⽂件的起始位置

8.1 fseek函数

该函数的作用是文件指针stream指向距origin(起始位置)偏移量为offset的位置 

origin有3中选择:SEEK_SET,SEEK_CUR,SEEK_END三种

 示例代码

int main()
{FILE* pf = fopen("test.txt", "r");if (pf == NULL){perror(fopen);return 0;}char a, b, c;fseek(pf, 5, SEEK_SET);a = fgetc(pf);printf("%c ", a);fseek(pf, 2, SEEK_CUR);b = fgetc(pf);printf("%c ", b);fseek(pf, -2, SEEK_END);c = fgetc(pf);printf("%c ", c);fclose(pf);return 0;
}

 test.txt文件内容

运行结果:

8.2 ftell函数

功能是返回文件指针相对于文件起始位置的偏移量。

 示例代码

int main()
{FILE* pf = fopen("test.txt", "r");if (pf == NULL){perror(fopen);return 0;}fseek(pf, -1, SEEK_END);int ret = ftell(pf);printf("%d", ret);fclose(pf);return 0;
}

8.3 rewind函数

这个函数功能是让文件指针回到文件的起始位置

 示例代码

int main()
{FILE* pf = fopen("test.txt", "r");if (pf == NULL){perror(fopen);return 0;}fseek(pf, -1, SEEK_END);printf("%c ", fgetc(pf));rewind(pf);printf("%c ", fgetc(pf));fclose(pf);return 0;
}

9.错误检测 

为什么要有错误检查呢

fgetc 如果读取正常,返回的是读取到字符的ascIl码值,如果读取的过程中遇到文件末尾,或者发生错误,就返回EOF(-1)

fgets如果读取正常,返回的是存储读取到的字符串的字符数组的地址,如果读取的过程中遇到文件末尾,或者发生错误,返回NULL

所以我们是无法知道是因为什么原因文件读取结束的。

9.1 feof函数

该函数用于判断当⽂件读取结束的时候,判断是读取结束的原因是否是:遇到⽂件尾结束(即是否是正常结束)

当是因为遇到⽂件尾而读取结束时返回非0值,否则返回0

9.2 ferror函数

该函数判断读取文件的时候是否发生错误

读取文件的时候是否发生错误0,否则返回非0值。

 示例代码

#include<stdio.h>
#include<errno.h>
int main()
{FILE* pf = fopen("test.txt", "r");if (pf == NULL){perror(fopen);return 1;}char c;while ((c = fgetc(pf)) != EOF){printf("%c", c);}if (feof(pf)){printf("\n遇到文件末尾,读取正常结束\n");}else if (ferror(pf)){perror(fgetc);}fclose(pf);return 0;
}

10.缓冲区

在认识缓冲区之前我们先想象一个场景,假设一辆公交车在司机在候车区看到1个人需要乘车就马上拉走一个,拉完再回来拉,每次几乎只拉一个人而路途又很长,那样的话可以想象效率是多么的低,解决方法就是公交车会在候车区停留30分钟到40分钟,等乘客足够多的时候一次性拉走,这样一来效率就高了很多。

像fgetc这样的函数是怎么实现把磁盘上的数据输入到内存呢,其实是依赖于操作系统提供的一些接口,fgetc,fgets,printf,sacnf......这些函数是调用了操作系统指令才得以实现的,如果每读写1个数据都调用一次操作系统,操作系统在执行的时候会被频繁的打断,效率变得很低。所以有了缓冲区的出现,让需要调用操作系统的数据先放在缓冲区,到最后一次性的执行。

数据:乘客

缓冲区:候车区

操作系统:公交车

10.1文件缓冲区

ANSI C标准采⽤“缓冲⽂件系统”处理的数据⽂件的,所谓缓冲⽂件系统是指系统⾃动地在内存中为程序中每⼀个正在使⽤的⽂件开辟⼀块“⽂件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才⼀起送到磁盘上。如果从磁盘向计算机读⼊数据,则从磁盘⽂件中读取数据输⼊到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的⼤⼩根据C编译系统决定的。

注意:关闭文件起到了刷新缓冲区的作用,这也是打开的文件一定要关闭的原因之一。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/297387.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java 包装类初识泛型

登神长阶 第六阶 包装类&初识泛型 目录 &#x1f600;一.包装类 &#x1f604;1.基本数据类型以及其对应的包装类 &#x1f602;2.装箱和拆箱 &#x1f607;2.1.装箱&#xff08;Boxing&#xff09; &#x1f609;2.2.拆箱&#xff08;Unboxing&#xff09; &#x…

【项目技术介绍篇】若依项目代码文件结构介绍

作者介绍&#xff1a;本人笔名姑苏老陈&#xff0c;从事JAVA开发工作十多年了&#xff0c;带过大学刚毕业的实习生&#xff0c;也带过技术团队。最近有个朋友的表弟&#xff0c;马上要大学毕业了&#xff0c;想从事JAVA开发工作&#xff0c;但不知道从何处入手。于是&#xff0…

55 npm run serve 和 npm run build 的分包策略

前言 这里我们来看一下 vue 这边 打包的时候的一些 拆分包的一些策略 我们经常会使用到 npm run build 进行服务的打包 然后 打包出来的情况, 可能如下, 可以看到 chunk-vendors 是进行了包的拆分, 我们这里就是 来看一下 这里 npm run build 的时候的, 一个分包的策略 测试…

【Linux实验室】NFS、DHCP的搭建

NFS、DHCP的搭建 1、nfs服务搭建及测试什么是NFS&#xff1f;环境准备服务端机器安装nfs-utils和rpcbind包启动NFS服务创建/data/NFSdata目录&#xff0c;配置nfs文件启动服务挂载测试在服务端在共享目录下创建文件测试在客户端在共享目录下创建文件 2、dhcp服务搭建及测试什么…

如何保护IP地址不被泄露?

当互联网成为每个家庭的重要组成部分后&#xff0c;IP地址就成了你的虚拟地址。您的请求从该地址开始&#xff0c;然后 Internet 将消息发送回该地址。那么&#xff0c;您担心您的地址被泄露吗&#xff1f; 对于安全意识高或者某些业务需求的用户&#xff0c;如果您正在寻找保护…

element-ui empty 组件源码分享

今日简单分享 empty 组件的源码实现&#xff0c;主要从以下三个方面&#xff1a; 1、empty 组件页面结构 2、empty 组件属性 3、empty 组件 slot 一、empty 组件页面结构 二、empty 组件属性 2.1 image 属性&#xff0c;图片地址&#xff0c;类型 string&#xff0c;无默认…

Facebook轮播广告是什么?投放过程中有哪些需要注意的吗?

轮播广告是Facebook广告形式中的一种&#xff0c;可以把3—5个广告合并到一个可滚动的广告单元中。轮播广告会出现在新鲜事即News Feed中&#xff0c;是独立站卖家常用的一种广告形式 为什么选择轮播广告&#xff1f; 转化率更高&#xff1a;相较于单图广告&#xff0c;轮播广…

LeetCode-热题100:2. 两数相加

题目描述 给你两个 非空 的链表&#xff0c;表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的&#xff0c;并且每个节点只能存储 一位 数字。 请你将两个数相加&#xff0c;并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外&#xff0c;这两个数都…

Day43 动态规划 part05

Day43 动态规划 part05 1049.最后一块石头的重量II 我的思路: 提示说和划分两个和相等的子集差不多&#xff0c;猛然想到&#xff0c;这道题不就是划分子集&#xff0c;用sum - 和最大*2 代码就是划分和相同的子集的变形 解答&#xff1a; class Solution {public int last…

Cache多核之间的一致性MESI

快速链接: 【精选】ARMv8/ARMv9架构入门到精通-[目录] &#x1f448;&#x1f448;&#x1f448; 思考: 1、为什么要学习MESI协议&#xff1f; 哪里用到了&#xff1f;你确定真的用到了&#xff1f; 2、MESI只是一个协议&#xff0c;总得依赖一个硬件去执行该协议吧&#xff0c…

蓝桥杯 --- 日期问题模板

目录 1.如何判断闰年 2.如何遍历当前年份的每一天 3.如果想要输出某一年某一天到某一年某一天之间一共有多少天。 4.精确到具体周几到周几的问题分析 5.如何直接通过一层for循环枚举年月日 习题&#xff1a; 蓝桥杯竞赛特别喜欢考日期问题&#xff0c;今天给大家分享一下…

VMware虚拟机安装Linux教程

以管理员身份运行VMware 按一下win键不然显示不全 重启即可。

什么是智慧公厕?智慧旅游下的智慧公厕功能和特点

智慧旅游下的智慧公厕功能和特点&#xff1f;智慧旅游是景区、公园、游乐场、文化场馆等领域的一种信息化解决方案&#xff0c;智慧公厕是智慧旅游极为重要的一部分&#xff0c;能大大提升游客满意度。智慧公厕采用物联网、互联网、大数据、云计算等技术&#xff0c;实现旅游景…

【Spring实战项目】SpringBoot3整合WebSocket+拦截器实现登录验证!从原理到实战

&#x1f389;&#x1f389;欢迎光临&#xff0c;终于等到你啦&#x1f389;&#x1f389; &#x1f3c5;我是苏泽&#xff0c;一位对技术充满热情的探索者和分享者。&#x1f680;&#x1f680; &#x1f31f;持续更新的专栏《Spring 狂野之旅&#xff1a;从入门到入魔》 &a…

picGo图床搭建gitee和smms(建议使用)

picGoGitee 这个需要下载gitee插件, 因为官方频繁的检索文件类型, 有时候也会失效 如果没有特殊要求平时存个学习的要看图中文字的重要的图片建议就是smms, 免费也够用! 图片存本地不方便, 各种APP中来回传还会失帧损失画质, 所以你值得往下看 picGosmms 建议使用这个, sm…

Linux gcc day3

find命令&#xff08;importance&#xff09;&#xff1a; 语法&#xff1a;find pathname -options find /root -name test.c which命令&#xff1a; which [指令] 只搜索指令&#xff0c;在什么位置下 为什么文件夹带有颜色呢&#xff1f; 科普补充alias命令&#xff1a; ali…

银行数字化转型导师坚鹏:银行数字化转型给分行带来的8大价值

银行数字化转型给分行带来的8大价值 银行数字化转型对不仅对总行产生了深远影响、给总行带来了新质生产力&#xff0c;对分行也会产生重要价值&#xff0c;银行数字化转型导师坚鹏从以下8个方面进行详细分析&#xff0c;相信能够给您带来重要启发&#xff0c;从而加速银行分行…

精读 Generating Mammography Reports from Multi-view Mammograms with BERT

精读&#xff08;非常推荐&#xff09; Generating Mammography Reports from Multi-view Mammograms with BERT&#xff08;上&#xff09; 这里的作者有个叫 Ilya 的吓坏我了 1. Abstract Writing mammography reports can be errorprone and time-consuming for radiolog…

clickhouse 源码编译部署

clickhouse 源码编译部署 版本 21.7.9.7 点击build project&#xff0c;编译工程&#xff0c;经过一定时间&#xff08;第一次编译可能几个小时&#xff0c;后续再编译&#xff0c;只编译有改动的文件&#xff09;生成release目录 在cmake-build-release → programs目录下…

Java集合(个人整理笔记)

目录 1. 常见的集合有哪些&#xff1f; 2. 线程安全的集合有哪些&#xff1f;线程不安全的呢&#xff1f; 3. Arraylist与 LinkedList 异同点&#xff1f; 4. ArrayList 与 Vector 区别&#xff1f; 5. Array 和 ArrayList 有什么区别&#xff1f;什么时候该应 Array而不是…