C语言的文件操作(炒详解)

news2024/11/25 5:22:01

⭐回顾回顾文件操作的相关细节⭐

欢迎大家指正错误

        📝在之前的学习中,不管增加数据,减少数据,当程序退出时,所有的数据都会销毁,等下次运行程序时,又要重新输入相关数据,如果一直像这样不能保持相关程序的数据会非常难受,我们想要把数据记录下来,只有我们们想删除数据的时候数据才会销毁,这就涉及了数据持久化,利用文件操作函数,我们可以将数据放在文件之中,下次需要使用可以直接访问

        🔥我们一般的数据持久化的方式就是把数据放在磁盘文件中,使用文件我们可以将数据直接存放到电脑的硬盘上,以做到数据的持久化。

目录

文件的基本操作

文件指针

文件的打开和关闭

利用"w"介绍相对路径

文件的读写

fputc函数

fgetc函数

fputs函数

fgets函数

fprintf函数和fscanf函数

sprintf和sscanf函数

二进制的读写函数

文件的定位

fseek函数

rewind函数

ftell函


📜介绍一下文件名

一个文件要有一个唯一的文件标识,以便用户识别和使用。

文件名包括三部分:

文件路径+文件名主干+文件后缀

例如:C:\code\test.txt

文件的路径的讲解

        我们传过去的路径有两种

①绝对路径

例如 "D:\桌面\planegames_boxed.exe"

         "C:\Users\Public\Videos"

👉在这里要注意哦,如果这样直接传地址会有转义字符的影响的。

在传参时,尽量在每个斜杠前加一个斜杠,就可以解决转义字符可能带来的影响

②相对路径

💭下边将会利用文件读写时的操作进行介绍更易理解


文件的基本操作

👉文件的基本操作包括文件的打开与关闭,除了标准的输入输出文件外,其他所有的文件都必需先打开再使用,使用后还必须关闭该文件。

文件指针

        📌文件指针是一个指向文件有关信息的指针,这些信息通常包括文件名,状态和当前的位置,他们保存在一个结构体变量中,在使用文件时需要在内存中为其分配空间,用来存放文件的基本信息,该结构体类型是系统定义的,C语言规定该类型为FILE

        不同的C编译站的FILE类型包括的内容完全不同,但是大同小异,这里的细节我们不必关心。

        一般都是通过创建一个FILE的指针来维护这个FILE结构的变量,这样使用起来可以更加方便。

创建一个FILE*类型的指针变量:

FILE * pf;

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

文件的打开和关闭

        文件在读写之前应该先打开文件,在使用结束之后应该关闭文件

(在程序结束前应该关闭所有的文件,目的是为了防止应为没有关闭文件而造成的数据流失。)

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

打开文件

🏅FILE * fopen(const char* filename(文件名),const char *mode(打开方式));

关闭文件

🏅int fclose (FILE*stream);

打开方式如下

👉用只读方式打开

通过上图来看,如果文件夹中没有该文件,就会返回一个空指针,用perror判断运行后如图,当然,如果创建了data.txt文件,就不会报错啦。


用写方式呢举一个梨子

int main()
{
	FILE* pfile;
	pfile = fopen("example","w");
	if (pfile == NULL)
	{
		perror("fopen");
		return 1;
	}
	if (pfile != NULL)
	{
		fputs("fopen example", pfile);
		fclose(pfile);
	}


	return 0;
}

        上面的表格我们注意到,用写的方式打开,如果没有文件会生成一个文件,执行相关指令。如果要打开的文件时绝对路径(例如桌面),没有该文件的话也还是会在桌面创建一个出来使用的。

“fputs后续我们会讲,是将字符串输入进文件中去。”

📜运行结果如图

一定要记得关闭程序前关闭所有的文件。


利用"w"介绍相对路径

📝前边加上一个点是当前目录,可以省略。

省略时:

📝删除创建好的lalala,不省略时再次运行

📝我们如果想在上一级目录里面创建lalala呢?

📝在前边加上(.),到达上一级,在上一级的x64里创建lalala

⭐如果还想往前跑,就继续加(.),在文件夹内找到要放置进去的文件夹,这就是文件的相对路径,利用.和\来找到具体的位置,相比较没有绝对路径那么精确。

文件的读写

        打开文件后,就可以进行文件的读写,C语言提供了丰富的文件操作函数,现在对其诸个介绍。以下所有函数默认FILE * fp

fputc函数

ch = fputc (ch,fp);

该函数的作用是把一个字符写进磁盘文件fp中,其中ch就是要输入的数据。fp是文件指针变量,如果函数输出成功,返回的就是输出的字符,如果输出失败,就返回EOF。

⭐看个例子

int main()
{
	FILE* fp;
	char ch;
	if ((fp = fopen("file", "w")) == NULL)
	{
		assert("fopen");
	}
	ch = getchar();
	while (ch != '#')
	{
		fputc(ch, fp);
		ch = getchar();
	}
	fclose(fp);
	return 0;
}

⭐运行后如图:

📑这个例子读取到#则停止。

fgetc函数

ch = fgetc (fp);

这个函数的作用是从指定文件(fp指向的文件)读取一个字符赋予ch。需要注意的是,文件必须是读或者读写的方式打开。

💡上面我们将file文件里写进了几个字符,现在我们来取出他们

int main()
{
	FILE* fp;
	char ch;
	fp = fopen("file", "r");
	ch = fgetc(fp);
	while (ch != EOF)
	{
		putchar(ch);
		ch = fgetc(fp);
	}
	fclose(fp);

	return 0;
}

💡运行后如图,可以发现确实将文件中写入的字符全部拿到了。

fputs函数

fputs(字符串,文件指针);

        和fputc不同的是,这个函数的作用是向指定文件中写入一个字符串,其中字符串可以是字符常量,也可以是字符数组名,指针或者变量。

💭看例子

int main()
{
	FILE* fp;
	char filename[30], str[30];
	printf("输入文件名\n");
	scanf("%s", filename);
	if ((fp = fopen(filename, "w")) == NULL)
	{
		perror("fopen");
	}
	printf("输入字符串\n");
	getchar();
	gets(str);
	fputs(str, fp);
	fclose(fp);
	return 0;
}

💡运行后如图

fgets函数

fgets(字符数组名,n,文件指针)

        该函数的作用是从指定文件中读一个字符串到字符数组中,n表示所得到的字符串中字符的个数(包含字符"\0")

要知道在上边我们在filecom里写进了hello world!

int main()
{
	FILE* fp;
	char str[30];
	if ((fp = fopen("filecom", "r")) == NULL)
	{
		perror("fopen error");
	}
	fgets(str, 11, fp);
	printf("%s", str);
	return 0;
}

💭运行后如图

上面所说包含"\0",在这里我们打印11个字符,然而这里直有10个字符(包含空格),这是因为"\0"也占了一个字符位。


fprintf函数和fscanf函数

        我们对printf函数和scanf应该都已经很熟悉了,下面要讲解的fprint和fscnaf与他们的作用相似,他们最大的区别就是读写的对象不同,fprintf和fscanf函数读写的对象不是终端,而是磁盘文件。

我们在cplusplus官网比较一番

        相比较printf函数,fprintf多了一个参数,fprintf函数形式如下

fprintf(fp,"%d",i);

        作用是将整型变量i的值以"%d"的格式输出到fp指向的文件中

💭举个梨子

int main()
{
	FILE* fp;
	int i = 666;
	if ((fp = fopen("filenum", "w")) == NULL)
	{
		perror("fopen error");
	}
	fprintf(fp, "%d", i);


	return 0;
}

创建一个filenum文件,写入666;

运行结果如下:

如果是%c写入呢,就要参考参照ASCII码表


fscanf函数

格式如下

fscanf(文件类型指针,格式字符串,输入列表);

fscanf(fp , "%d" , &i);

⭐我们先写入abcdefg

💡然后以字符的形式输出

运行后结果正常,输出为参照ASCII对应的数字。


sprintf和sscanf函数

前边已经了解了fprintf和fscanf函数

        💡再对比前边的fprintf函数,相比于fprintf函数将内容写进文件中,可以发现sprintf函数的第一个参数变成了一个字符指针,sprintf的作用即是将格式化的数据转化成字符串,放在传进来的字符数组里。

举一个例子

typedef struct Nums
{
	int a;
	char b;
	double c;
}Nums;

int main()
{
	char str[30] = "0";
	Nums nums = { 3,'f',1.5 };
	sprintf(str, "%d %c %lf\n", nums.a, nums.b, nums.c);
	printf("%s", str);
	return 0;
}

仔细观擦发现和printf函数差不多,作用也很相似。

sscanf同理

💡sscanf从字符串中读取格式化的数据。

typedef struct Nums
{
	int a;
	char b;
	double c;
}Nums;

int main()
{
	char str[30] = "0";
	Nums nums = { 3,'f',1.5 };
	Nums nums1 = { 0 };
	sprintf(str, "%d %c %lf\n", nums.a, nums.b, nums.c);
	//printf("%s", str);
	sscanf(str, "%d %c %lf", &(nums1.a), &(nums1.b), &(nums1.c));
	printf("%d %c %lf", nums.a, nums.b, nums.c);
	return 0;
}

📖读取str内的元素放进nums1中去,此时再打印结构体变量nums1就会发现已经把str内的数据搬进nums1中啦。


二进制的读写函数

⭐前边所介绍的fputc和fgetc函数,每次只能读写文件的一个字符,但我们在编写程序的过程中常常需要对整块数据进行读写,例如,对一个结构体类型的变量值进行读写,下面进行fread和fwrite函数。

因为写进去的是二进制文件,所以当我们用记事本打开时,看到的都是乱码。

代码如下

typedef struct Nums
{
	int a;
	char b;
	double c;
	char str[10];
}Nums;
int main()
{
	Nums nums = { 1,'c',1.5,"heihei"};
	FILE* pf = fopen("data.txt", "wb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fwrite(&nums, sizeof(Nums), 1, pf);
	fclose(pf);
	return 0;
}

运行后如下

        👑接下来我们要将它使用fread再读出来,放进一个结构体中,再将其打印出来,将上边的代码改造一下

📜代码如下

typedef struct Nums
{
	int a;
	char b;
	double c;
	char str[10];
}Nums;
int main()
{
	Nums nums = { 0 };
	FILE* pf = fopen("data.txt", "rb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fread(&nums, sizeof(Nums), 1, pf);
	printf("%d %c %f %s\n", nums.a, nums.b, nums.c, nums.str);
	fclose(pf);
	return 0;
}

👉运行后代码如下,我们将nums已经置空了,但打印结果已经说明了一切。

文件的定位

⭐学习了前边的函数,我们这时候又要思考了,在对文件进行操作时,一定要从头开始吗?多不方便哇,这时候就需要文件定位函数来实现对文件的随机读取。

fseek函数

fseek(文件类型指针,位移量,起始点);

        这个函数的作用是移动文件内部的位置指针,其中,“文件类型指针”指向被移动的文件;“移动量”表示移动的字节数,要求位移量是long类型数据。“起始点”表示从何处开始计算位移量,规定的起始点有文件首,当前位置,文件末。

表示方法如图

🙉怎么用呢?

fseek(fp,-20,1);

fseek(fp,-20,SEEK_CUR);

        代码表示将位置指针从当前位置向后退20个字节。

🙉看代码

int main()
{
	FILE* file = fopen("data.txt", "w");
	fputs("This is an apple.", file);
	fseek(file, 9, 0);
	fputs(" sam", file);
	fclose(file);
	return 0;
}

⭐运行后如图

        结合运行结果很容易发现,在替换时将空格也替换了,输入的位移量是九,在输入是从第十个位置继续输入,然后puts里的字符串覆盖原字符串。

fseek(fp,5,0);

       💡 此代码的含义是将文件指针指向距离文件首5个字节的位置,也就是指向字符串中的第六个字符。

rewind函数

💭前边讲过了fseek函数,这里介绍的rewind函数也可以起到定位文件指针的作用

int rewind(文件类型指针);

该函数的作用是使位置指针重新返回文件的开头,该函数没有返回值。

举一个例子

int main()
{
	FILE* fp;
	char ch;
	fp = fopen("data.txt", "r");
	
	ch = fgetc(fp);
	while (ch != EOF)
	{
		putchar(ch);
		ch = fgetc(fp);
	}
	rewind(fp);
	ch = fgetc(fp);
	while (ch != EOF)
	{
		putchar(ch);
		ch = fgetc(fp);
	}
	fclose(fp);
	return 0;
}

运行结果如图所示

有点懵?再来看一个例子

int main()
{
	int n;
	FILE* pfile;
	char buffer[27];
	pfile = fopen("myfile.txt", "w+");
	for (n = 'A'; n <= 'Z'; n++)
	{
		fputc(n, pfile);
	}
	//rewind(pfile);
	fread(buffer, 1, 26, pfile);
	fclose(pfile);
	buffer[26] = '\0';
	puts(buffer);

	return 0;
}

可以看到将rewind注释掉后,运行结果如下,这是因为在使用fputc时将文件指针移动到了最后,再读的话就是从最后的位置开始读,所以输出结果为空。

如果将rewind解注释,再次运行

与上次运行的结果不同,此时将文件指针重新返回文件的开头,该函数没有返回值。

ftell函数

        嘿嘿嘿,如果上边解释rewind大家还有点不懂,可以结合ftell函数来解释哦!

ftell函数一般形式如下

long ftell(文件类型指针)

        该函数的作用是返回文件指针相对于起始位置的偏移量。

⭐利用同样的实例,来看一下是否rewind函数真的把文件指针搞到了最前边。同样也可以摸清ftell函数的作用。

int main()
{
	int n;
	int size = 0;
	FILE* pfile;
	char buffer[27];
	pfile = fopen("myfile.txt", "w+");
	for (n = 'A'; n <= 'Z'; n++)
	{
		fputc(n, pfile);
	}
	size = ftell(pfile);//看一下此时的文件指针的位置
	printf("%d\n", size);
	rewind(pfile);//指针退回到开头位置
	size = ftell(pfile);
	printf("%d\n", size);//再看一次
	fread(buffer, 1, 26, pfile);
	fclose(pfile);
	buffer[26] = '\0';
	puts(buffer);

	return 0;
}

运行后如图

我想已经很明显啦,到了这里文件操作相关的知识点就梳理完毕啦,如果有错误欢迎大家指出!

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

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

相关文章

升级OpenSSL并进行编译安装

Packaging (OpenSSL)组件存在安全漏洞的原因是由于当前爆出的Openssl漏洞。 这个漏洞可能会导致泄露隐私信息&#xff0c;并且涉及的机器和环境也有所不同&#xff0c;因此修复方案也会有所不同。 目前&#xff0c;一些服务器使用的Nginx是静态编译OpenSSL&#xff0c;直接将Op…

【数据结构】C++实现红黑树

【数据结构】C实现红黑树 红黑树的概念 红黑树&#xff0c;是一种二叉搜索树&#xff0c;但在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是Red或Black。 通过对任何一条从根到叶子的路径上各个结点着色方式的限制&#xff0c;红黑树确保没有一条路径会比其他路…

多数网工碌碌无为,都是败在这件事上

大家好&#xff0c;我是老杨。 这周是网络安全周&#xff0c;博览会的现场很热闹&#xff0c;我也被邀请去参观了一圈。 主要是学习观摩各大厂出的最新的安全产品、解决方案等。 这几天&#xff0c;也遇上了不少大佬。 有从路由交换转到安全的&#xff0c;也有从渗透转到防御…

Unity中Shader的模板测试

文章目录 前言什么是模板测试1、模板缓冲区2、模板缓冲区中存储的值3、模板测试是什么&#xff08;看完以下流程就能知道模板测试是什么&#xff09;模板测试就是在渲染&#xff0c;后渲染的物体前&#xff0c;与渲染前的模板缓冲区的值进行比较&#xff0c;选出符合条件的部分…

红队打靶:Me and My Girlfriend打靶思路详解(vulnhub)

目录 写在开头 第一步&#xff1a;主机发现和端口扫描 第二步&#xff1a;Web渗透&#xff08;修改XFF代理&#xff09; 第三步&#xff1a;数据库手工枚举 第四步&#xff1a;sudo php提权 总结与思考 写在开头 本篇博客在自己的理解之上根据大佬红队笔记的视频进行…

GPIO子系统编写LED灯的驱动、linux内核定时器

一、GPIO子系统 1.概念&#xff1a; 一个芯片厂商生产出芯片后会给linux提供一个当前芯片中gpio外设的驱动&#xff0c;我们当前只需要调用对应的厂商驱动即可完成硬件的控制。而linux内核源码中的gpio厂商驱动有很多&#xff0c;这里linux内核对厂商驱动做了一些封装&#x…

基于Java的大学生心理健康答题小程序设计与实现(亮点:选题新颖、可以发布试卷设置题目、自动判卷、上传答案、答案解析)

校园点餐小程序 一、前言二、我的优势2.1 自己的网站2.2 自己的小程序&#xff08;小蔡coding&#xff09;2.3 有保障的售后2.4 福利 三、开发环境与技术3.1 MySQL数据库3.2 Vue前端技术3.3 Spring Boot框架3.4 微信小程序 四、功能设计4.1 主要功能描述 五、系统主要功能5.1 登…

python-字符串去掉空格的常见方法

python提供了去掉字符串空格的方法&#xff0c;可以满足大部分需求。 但在实际应用中&#xff0c;还需要灵活借助python其他方法&#xff0c;来实现字符串空格的删除。 比如&#xff0c;去掉字符串的全部空格、字符串连续空格保留一个等&#xff0c;都需要结合其他的方法来实现…

【kafka】mac环境安装kafka

生产中使用到的中间件使用心得记录&#xff0c;感兴趣可以关注我一起学习&#xff5e; 环境&#xff1a; 硬件&#xff1a;mac 软件&#xff1a;kafka v3.0.0 安装步骤&#xff1a; 终端安装&#xff1a; 通过brew安装&#xff0c;会默认直接安装最新的版本 步骤1&#xf…

Android StringFog 字符串自动加密

一、StringFog 作用 一款自动对dex/aar/jar文件中的字符串进行加密Android插件工具&#xff0c;正如名字所言&#xff0c;给字符串加上一层雾霭&#xff0c;使人难以窥视其真面目。可以用于增加反编译难度&#xff0c;防止字符串代码重复。 支持java/kotlin。支持app打包生成…

【JDK 8-函数式编程】4.6 方法引用与构造函数引用

一、 方法引用与构造函数引用 1. 说明 2. 语法: 二、静态方法 三、实例方法 四、构造函数 4.1 单个参数 4.2 2个参数 五、执行结果 一、 方法引用与构造函数引用 以前调用&#xff1a;对象.方法名、类名.方法名 jdk1.8提供了另外一种调用方式 :: 1. 说明 用来直接访…

子网的划分

强化计算机网络发现王道没有这一块的内容&#xff0c;导致做题稀里糊涂。于是个人调研补充。 子网划分是将一个大型IP网络划分成更小的子网&#xff0c;以实现更有效的网络管理和资源分配。 原因&#xff1a; 提高网络性能&#xff1a;子网划分可以减少广播域的大小&#xff…

靶场练习——SDcms文件上传漏洞靶场

文章目录 前言一、寻找网站后台页面1、点击请登录&#xff0c;查看URL2、修改URL参数&#xff0c;找到后台登录页面 二、登录后台管理系统1、不能使用爆破2、使用弱口令登录 三、寻找文件上传点四、上传文件操作1、上传普通的图片文件&#xff0c;查看数据包2、尝试上传PHP文件…

Mysql----锁

文章目录 锁 概述 全局锁全局锁 概述全局锁 操作表级锁表级锁 表锁表级锁 元数据锁表级锁 意向锁行级锁行级锁 行锁行级锁 间隙锁&临键锁 锁 概述 是什么 是计算机协调多个进程或线程并发访问某一资源的机制。 意义 在数据库中&#xff0c;数据是一种供许多用户共享的…

C/C++计算分数的浮点数值 2019年12月电子学会青少年软件编程(C/C++)等级考试一级真题答案解析

目录 C/C计算分数的浮点数值 一、题目要求 1、编程实现 2、输入输出 二、解题思路 1、案例分析 三、程序代码 四、程序说明 五、运行结果 六、考点分析 C/C计算分数的浮点数值 2019年12月 C/C编程等级考试一级编程题 一、题目要求 1、编程实现 两个整数a和b分别作…

每日一题 337. 打家劫舍 III

难度&#xff1a;中等 整体思路相当于是前两天的方法倒过来&#xff0c;毕竟二叉树最常用的解法就是递归倒推 对于每一颗子树&#xff0c;他必定有一种最大的盗取方法&#xff0c;但是只有它的 root 的盗取情况才会影响到 root 的父节点&#xff0c;即如果收益最大的盗取方法…

【TCP】滑动窗口、流量控制 以及拥塞控制

滑动窗口、流量控制 以及拥塞控制 1. 滑动窗口&#xff08;效率机制&#xff09;2. 流量控制&#xff08;安全机制&#xff09;3. 拥塞控制&#xff08;安全机制&#xff09; 1. 滑动窗口&#xff08;效率机制&#xff09; TCP 使用 确认应答 策略&#xff0c;对每一个发送的数…

机器学习第七课--情感分析系统

分词 分词是最基本的第一步。无论对于英文文本&#xff0c;还是中文文本都离不开分词。英文的分词相对比较简单&#xff0c;因为一般的英文写法里通过空格来隔开不同单词的。但对于中文&#xff0c;我们不得不采用一些算法去做分词。 常用的分词工具 # encodingutf-8 import …

Linux C 多线程

为什么会有线程? ————————>>>> 进程实现多任务的缺点&#xff1a; 进程间切换的计算机资源开销很大&#xff0c;切换效率非常低进程间数据共享的开销也很大 线程和进程的关系 线程是进程的一个执行单元&#xff0c;是进程内的调度实体。比进程…

应用:使用#todo标签,清理未完成笔记 | Obsidian实践

各位【见睿思齐】的朋友&#xff0c;好久不见。断更了这么久&#xff0c;非常抱歉。起初是因为家里出了些事情&#xff0c;不得不断更了一段时间&#xff1b;后来是因为断更太久&#xff0c;找不回写作的感觉&#xff0c;于是又断更了更久的时间……总之是太不应该。最近读了一…