C语言文件操作【超详解】

news2024/11/26 4:52:27

目录

1.为什么使用文件

2.什么是文件

2.1程序文件

2.2数据文件

2.3文件名

3.文件的打开和关闭

3.1文件指针

3.2文件的打开和关闭

3.2.1fopen函数

3.2.2fclose函数 

4.文件的顺序读写

4.1顺序读写函数介绍

4.1.1 fputc、fgetc和fputs、fgets

4.1.2fprintf、fscanf(格式化读写)

 4.1.3fread、fwrite(二进制读写)

5.文件的随机读写

5.1 fseek

5.2 ftell

5.3 rewind 

6.文本文件和二进制文件

7.文件读取结束的判定

7.1被错误使用的feof

8.文件缓冲区


1.为什么使用文件

我们前面学习结构体时,写通讯录的程序,当通讯录运行起来的时候,可以给通讯录中增加、删除数
据,此时数据是存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯
录程序的时候,数据又得重新录入,如果使用这样的通讯录就很难受。
我们在想既然是通讯录就应该把信息记录下来,只有我们自己选择删除数据的时候,数据才不复存在。
这就涉及到了数据持久化的问题,我们一般数据持久化的方法有,把数据存放在磁盘文件、存放到数据
库等方式。
使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。

2.什么是文件

磁盘上的文件是文件。
但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件(从文件功能的角度来分类的)

2.1程序文件

包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境
后缀为.exe)

2.2数据文件

文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,
或者输出内容的文件。

这篇文章讨论的是数据文件

2.3文件名

一个文件要有一个唯一的文件标识,以便用户识别和引用。
文件名包含3部分:文件路径+文件名主干+文件后缀
例如: c:\code\test.txt
为了方便起见,文件标识常被称为文件名。

3.文件的打开和关闭

3.1文件指针

缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”。
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型由系统声明的,取名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;

这是vs2022编译环境提供的 stdio.h 头文件中有以下的文件类型声明:

    typedef struct _iobuf
    {
        void* _Placeholder;
    } FILE;

不同的编译器的FILE类型包含的内容不完全相容,但是大同小异。

每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。

下面我们可以创建一个 FILE* 的指针变量 :

FILE* pf ;//文件指针变量

定义pf是一个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区(是一个结构体变量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够找到与它关联的文件。

比如:

3.2文件的打开和关闭

文件在读写之前应该先打开文件,在使用结束之后应该关闭文件。
在编写程序的时候,在打开文件的同时,都会返回一个FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系。

ANSIC规定使用fopen函数来打开文件,fclose函数来关闭文件。

3.2.1fopen函数

通过查询我们可知 fopen函数的返回值为FILE*(文件指针),两个const修饰的参数,第一个参数

const char  * filename  为我们所要进行操作的文件的文件名,第二个参数  const char * mode 为我们要对文件进行怎样的操作。

3.2.2fclose函数 

通过查询我们可知fclose 函数返回值为int ,参数FILE * stream 的意思是文件指针指向的流。

打开方式如下:

实例:

int main()
{
	//打开文件
	FILE* pf = fopen("ceshi.txt", "r");
	//判断是否打开成功
	if (pf == NULL)
	{
		perror("fopen");//没有打开成功将错误码转化为错误信息在屏幕上打印
		return 1;
	}
	//对文件进行操作



	//关闭文件  将文件指针置空
	fclose(pf);
	pf = NULL;
	return 0;
}

文件打两种开路径

在打开文件时我们基本打开的是代码所在的同一级文件夹这种打开方式叫相对路径打开 

如果我们要打开其他路径下的文件就必须加上完整的路径(注意转移字符)这种打开方式叫绝对路径打开

4.文件的顺序读写

4.1顺序读写函数介绍

 4.1.1 fputc、fgetc和fputs、fgets

fputc函数用来对文件进行 ”w“ (只写)返回值为 int 类型,两个参数为 int 类型的字符(ASCII值) 和文件指针。

实例:

int main()
{
	FILE* pf = fopen("data.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputc('h', pf);
	fputc('e', pf);
	fputc('l', pf);
	fputc('l', pf);
	fputc('o', pf);

	fclose(pf);
	pf = NULL;
	return 0;
}

将上述代码运行后在我们代码的同一级目录会产生一个 data.txt 的文件 

 

fgetc函数用来对文件进行 ”r“(只读)返回值为 int ,参数文件指针

实例:

int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//读文件
	//int ch = fgetc(stdin);
	int ch = 0;
	printf("%c", ch);
	ch = fgetc(pf);
	printf("%c", ch);
	ch = fgetc(pf);
	printf("%c", ch);
	ch = fgetc(pf);
	printf("%c", ch);
	ch = fgetc(pf);
	printf("%c", ch);
	ch = fgetc(pf);
	printf("%c", ch);
	fclose(pf);
	pf = NULL;
	return 0;
}

 

fputs和fputc函数最大的区别就是fputs的第一个参数变为了字符串。

实例:

int main()
{
	FILE* pf = fopen("data.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//fputs("hello world\n", stdout);
	fputs("hello\n", pf);
	fputs("world\n", pf);

	fclose(pf);
	pf = NULL;
	return 0;
}

 

 

该函数从指定的文件中读取最多n-1个字符的字符串,并将其存储在str指向的字符数组中。它会读取换行符之前的字符,包括换行符本身,并在读取完毕后,在最后一个字符后面自动添加一个空字符('\0');

实例:让我们对刚才的文件使用fgets

4.1.2fprintf、fscanf(格式化读写)

 实例:

struct s
{
	int a;
	float b;
};
int main()
{

	struct s sa = { 100,3.14 };
	FILE* pf = fopen("geshihua.txt", "w");
	if (pf == NULL) 
	{
		perror("fopen");
		return 1;
	}
	fprintf(pf, "%d %f", sa.a, sa.b);

	fclose(pf);
	pf = NULL;
	return 0;
}

 

实例:

struct s
{
	int a;
	float b;
};
int main()
{

	struct s sa = { 0 };
	FILE* pf = fopen("geshihua.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");

		return 1;
	}

	fscanf(pf, "%d %f", &(sa.a), &(sa.b));
	printf("%d %f", sa.a, sa.b);
	fclose(pf);
	pf = NULL;
	return 0;
}

 4.1.3fread、fwrite(二进制读写)

 实例:

struct s
{
	int a;
	float b;
	char c[20];
};
int main()
{
	struct s sa = { 100,3.14,"hello world" };
	FILE* pf = fopen("erjinzhi.txt", "wb");必须使用wb

	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fwrite(&sa, sizeof(struct s), 1, pf);


	fclose(pf);
	pf = NULL;
	return 0;
}

 

 实例:

struct s
{
	int a;
	float b;
	char c[20];
};
int main()
{

	struct s sa = { 0 };
	FILE* pf = fopen("erjinzhi.txt", "rb");必须使用rb
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fread(&sa, sizeof(struct s), 1, pf);
	printf("%d %f %s", sa.a, sa.b, sa.c);
	fclose(pf);
	pf = NULL;
	return 0;
}

5.文件的随机读写

5.1 fseek

设置文件指针位置的函数。它用于在文件中移动文件指针到指定位置,以便进行读取或写入操作。

实例:

int main()
{
	FILE*pf=fopen("data.txt", "r");//hello world
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//定位文件指针到f

	fseek(pf, 4, SEEK_SET);

	int ch = fgetc(pf);
	printf("%c\n", ch);
	fclose(pf);
	pf = NULL;
	return 0;
}

我们在上述data.txt的文件中存储着 hello world字符串,这个程序刚开始的文件指针指向使用fseek函数使其移动到o的位置,在使用fgetc函数将其读取出来。

5.2 ftell

用于获取文件指针的当前位置的函数。它返回一个long类型的值,表示文件指针相对于文件起始位置的偏移量(以字节为单位)。

实例:

int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int ch = fgetc(pf);
	printf("%c ", ch);
	ch = fgetc(pf);
	printf("%c ", ch);
	ch = fgetc(pf);
	printf("%c ", ch);
	ch = fgetc(pf);
	printf("%c ", ch);

	int i=ftell(pf);

	printf("%d ", i);
	fclose(pf);
	pf = NULL;
	return 0;
}

当我们使用fgetc函数获取文件中的一个字符时,文件内部的文件指针就会移动一位,ftell函数就是获取当前文件指针的偏移量的。

5.3 rewind 

用于将文件指针重新设置到文件的起始位置.

实例:

int main()

{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int ch = getc(pf);
	printf("%c\n", ch);
	ch = getc(pf);
	printf("%c\n", ch);
	ch = getc(pf);
	printf("%c\n", ch);
	ch = getc(pf);
	printf("%c\n", ch);

	rewind(pf);

	ch = getc(pf);
	printf("%c\n", ch);
	fclose(pf);
	pf = NULL;
	return 0;
}

 我们先使用fgetc函数获取文件的字符使文件指针偏移,在使用rewind函数回到起始位置。

6.文本文件和二进制文件

根据数据的组织形式,数据文件被称为文本文件或者二进制文件。
数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。
如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件。

一个数据在内存中是怎么存储的呢?
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。
如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占用5个字节(每个字符一个字节),而二进制形式输出,则在磁盘上只占4个字节。(vs2022测试)

测试案例:

#include <stdio.h>
int main()
{
	int a = 10000;
	FILE* pf = fopen("test.txt", "wb");
	fwrite(&a, 4, 1, pf);//二进制的形式写到文件中
	fclose(pf);
	pf = NULL;
	return 0;
}

我们会到代码的目录下打开那个文件发现我们看不懂,因为 这是二进制文件

 

 

7.文件读取结束的判定

7.1被错误使用的feof

牢记:在文件读取过程中,不能用feof函数的返回值直接来判断文件的是否结束。
feof 的作用是:当文件读取结束的时候,判断是读取结束的原因是否是:遇到文件尾结束。
1. 文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
例如:
        fgetc 判断是否为 EOF .
        fgets 判断返回值是否为 NULL .
2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
例如:
        fread判断返回值是否小于实际要读的个数。

实例:将一个文件中的所有文件复制到另个文件中。

int main()

{
	FILE* pfread = fopen("data1.txt", "r");
	if (pfread == NULL)
	{
		perror("fopen");
		return 1;
	}
	FILE* pfwrite = fopen("data2.txt", "w");
	if (pfwrite == NULL)
	{
		perror("fopen");
		fclose(pfread);
		pfread = NULL;
		return 1;
	}
	int ch = 0;
	while ((ch=fgetc(pfread)) != EOF)
	{
		fputc(ch, pfwrite);
	}
	fclose(pfread);
	pfread = NULL;
	fclose(pfwrite);
	pfwrite = NULL;
	return 0;
}

8.文件缓冲区

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

这里可以得出一个结论:
因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区或者在文件操作结束的时候关闭文件。
如果不做,可能导致读写文件的问题 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

装饰(Decorator)模式

目录 动机使用场景参与者协作效果实现相关模式应用和思考类图 装饰模式是对象结构型模式&#xff0c;动态地给一个对象添加一些额外的职责。就增加的功能来说装饰器模式相比生成子类更为灵活。 动机 有时候需要给某个对象而不是整个类添加一些功能&#xff0c;使用继承也可以&…

【SDOF振荡器的非线性-非弹性多轴时间响应分析】用于SDOF振荡器非线性非弹性时程分析的鲁棒性分析研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

经典指针笔试题你会了嘛

作者简介&#xff1a;დ旧言~&#xff0c;目前大一&#xff0c;现在学习Java&#xff0c;c&#xff0c;Python等 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 望小伙伴们点赞&#x1f44d;收藏✨加关注哟&#x1f495;&#x1f495; 指针和数组笔试题 &…

【Linux】多线程 之 POSIX信号量

文章目录 1. 概念2. 信号量的工作机制3. 认识接口sem_init ——初始化信号量sem_destroy——销毁信号量sem_wait ——申请信号量sem_post ——释放信号量 4. 基于环形队列的生产消费模型原理解析代码代码解析ringqueue类构造析构push ——生产pop ——消费 代码实现Ringqueue.h…

精通Skia引擎,发挥应用程序中高性能图形的潜力

Skia是由Google开发的一个开源的2D图形库。它提供了丰富的图形绘制、图像处理和渲染功能&#xff0c;适用于多个平台&#xff0c;包括操作系统、浏览器和移动设备等。Skia主要用于处理2D图形的绘制、文字渲染、图像处理和用户界面的呈现。 特点和功能&#xff1a; 高性能&…

基于Javaweb实现ATM机系统开发实战(十一)存储交易记录

首先创建一个业务接口&#xff1a; package com.atm.service;import com.atm.pojo.RunMessage;//交易记录的业务接口 public interface RunMessageService{//添加交易记录public void addRunMessage(RunMessage runMessage) throws Exception ; }再完成业务接口的实现类&#…

iNav开源代码之FAILSAFE

iNav开源代码之FAILSAFE 1. 源由2. FAILSAFE类别3. FAILSAFE配置4. FAILSAFE阶段&状态机5. 参考资料 1. 源由 最近因为炸机&#xff0c;百思不得其解。 关于炸鸡的过程&#xff0c;就不再展开&#xff0c;都是“泪”啊&#xff01;想进一步了解的&#xff0c;请参阅前面的…

科普一下Elasticsearch中BM25算法的使用

首先还是先了解几个概念&#xff0c;Elasticsearch是一个开源的分布式搜索和分析引擎&#xff0c;它使用一系列算法来计算文档的相关性分数&#xff08;relevance score&#xff09;。这些算法用于确定查询与文档的匹配程度&#xff0c;以便按相关性对搜索结果进行排序。以下是…

【C++】多线程编程一(初识并发和多线程)

目录 一、并发和并行的概念 1.并发 2.并行 3.并发和并行的区别 二、并发的途径 多进程并发 多线程并发 三、C11相关多线程编程的头文件 一、并发和并行的概念 1.并发 并发&#xff1a;指同一时刻只能有一条指令执行&#xff0c;但是多个进程指令被快速地轮换执行&#…

C语言--文件操作

文章目录 前言文件文件名 文件的打开和关闭文件指针文件的打开和关闭 文件的顺序读写fgetc、fputcfgets、fputsfprintf、fsanffread、fwrite 文件的随机读写fseekftellrewind 文本文件和二进制文件文件读取结束的判定feof 文件缓冲区 前言 在我们写完一个程序之后&#xff0c;…

【数据结构】树二叉树的概念以及堆的详解

⭐️ 往期相关文章 ✨链接1&#xff1a;【数据结构】顺序表 ✨链接2&#xff1a;【数据结构】单链表 ✨链接3&#xff1a;【数据结构】双向带头循环链表 ✨链接4&#xff1a;【数据结构】栈和队列 ⭐️ 树的概念 百度百科的解释&#xff1a;树是一种非线性的数据结构&#xf…

Jmeter四种关联方法讲解

目录 方法一&#xff0c;从前一个请求中取&#xff0c;用正则表达式提取器。 二、json path extractor 三、json extractor 四、XPath Extractor 方法一&#xff0c;从前一个请求中取&#xff0c;用正则表达式提取器。 具体方法&#xff0c;在需要获得数据的请求上右击添加…

【09】STM32·HAL库-新建HAL库版本MDK工程 | 下载STM32固件库

目录 1&#xff0c;新建工程前的准备工作&#xff08;了解&#xff09;1.1下载相关STM32Cube 官方固件包&#xff08;F1/F4/F7/H7&#xff09; 2&#xff0c;新建HAL库版本MDK工程步骤&#xff08;熟悉&#xff09;2.1新建工程文件夹2.1.1Drivers文件夹2.1.2Middlewares文件夹2…

基数排序(Radix_Sort)

基数排序 (Radix Sort)-20230715 前言 基数排序适用于多关键字排序&#xff0c;与前述的比较排序不同&#xff0c;实现基数排序不需要对关键字进行比较和移动。简而言之&#xff0c;基数排序是一类借助多关键字排序的思想对单逻辑关键字实现排序的方法。 多关键字排序 先看…

SwiftUI 原生或利用 Vision 检测限定高度的 Text 视图能否完整显示文本的方法

功能需求 在 SwiftUI 开发中,为了节省空间我们往往会为内容很长的文本视图(Text)限定一个高度,然后让用户决定是否展开显示其完整内容。 如上图所示,为了节省空间我们对 Text 视图高度做了限制,然后根据文本长度自动显示或隐藏展开按钮,用户点击该按钮即可展开显示完整…

FL Studio21入门版编曲 2023年免费小白新手编曲工具

全能数字音乐工作站&#xff08;DAW&#xff09;编曲、剪辑、录音、混音&#xff0c;有了它就能把你的笔记本电脑变成全功能音乐工作室。 内置丰富插件&#xff0c;满足不同风格创作拥有强大的采样引擎&#xff0c;自带高品质打击乐、钢琴、弦乐、吉他等107种乐器效果。 流行…

C++-把字符串转换成整数

题目来源&#xff1a;牛客网 题目描述&#xff1a; 将一个字符串转换成一个整数&#xff0c;要求不能使用字符串转换整数的库函数。 数值为 0 或者字符串不是一个合法的数值则返回 0 数据范围&#xff1a;字符串长度满足 0≤n≤100 进阶&#xff1a;空间复杂度 O(1) O(1) &…

Mybatis执行SQL过程

文章目录 1. 相关代码2. 创建SqlSession3. 创建Mapper代理对象4.sql的执行4.1 MapperProxy.invoke()4.2 mapperMethod.execute()4.3 sqlSession.selectOne4.4 CachingExecutor.query()4.5 BaseExecutor.query方法4.6 SimpleExecutor.doQuery方法 1. 相关代码 Testpublic void …

其他形式转欧拉角形式

1. 坐标系轴方向问题 3D数学基础中约定使用左手坐标系 左手坐标系 右手坐标系 左手正方向&#xff1a;x正向右平移&#xff0c;y向上平移&#xff0c;z向前平移. 右手正方向&#xff1a;x正向左平移&#xff0…

漫谈大数据时代的个人信息安全(二)——“逢脸造戏”

大数据时代的个人信息安全系列二&#xff1a;“逢脸造戏” 1. 逢脸造戏2. 生物识别信息安全2.1 生物识别信息被大量获取2.2 生物识别信息被非法滥用 3. 各国加强对深度合成监管4. 个人信息保护小贴士 互联网就像公路&#xff0c;用户使用它&#xff0c;就会留下脚印。 每个人都…