文件操作(下)(想要了解如何操作文件,那么看这一片就足够了!)

        前言:在文件操作(上)中,我们讲到了基础的文件操作,包括文件的打开,文件的关闭,以及文件的基础读写,那么除了之前学习的读写之外,还有什么其他的方式对文件进行读写操作吗?


✨✨✨这里是秋刀鱼不做梦的BLOG

✨✨✨想要了解更多内容可以访问我的主页秋刀鱼不做梦-CSDN博客

那么废话不多说,我们直接开始讲解,书接上回,我会从以下三个方面继续我们的讲解:

想看文件操作(上),点击------------------------------------------------------------------------------------------->文件操作(上)(想要了解如何操作文件,那么看这一片就足够了!)-CSDN博客

目录

1.读写文件的其他操作

        (1)fscanf函数

        (2)fprintf函数

        补充:

区分printf 、fprintf 、 sprintf函数  + 区分scanf 、fscanf 、sscanf函数

        (3)fwrite函数

        (4)fread函数

2.文件的随机读写

        (1)fseek函数

        (2)ftell函数

        (3)rewind函数

3.文件读取结束的判定

        (1)feof函数

        (2)ferror函数


1.读写文件的其他操作

对于文件操作还有以下的几个函数:

函数名功能适用性
fscanf()格式化输入函数所有输入流
fprintf()格式化输出函数所有输出流
fread()二进制输入文件
fwrite()二进制输出文件

让我们一个一个来分析:

        (1)fscanf函数

先看一下官网对其的解释:

该函数的第一个参数是一个文件指针,第二个参数format是格式字符串,用于指定要读取的数据的格式;第三个参数...是要读取的数据的地址,它的返回值为成功匹配和读取的数据项数量。

我们直接使用实例来加深对其的理解:

#include<stdio.h>
struct Student
{
	char name[20];
	int age;
};
int main()
{
	struct Student stu = { 0 }; 
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//使用fscanf函数格式化将文件中内容输出
	fscanf(pf, "%s %d", stu.name, &(stu.age));
	printf("%s %d", stu.name, stu.age);
	fclose(pf);
	pf = NULL;
	return 1;
}

这样我们就完成了数据的格式化输出。

        (2)fprintf函数

先看一下官网对其的解释:

该函数的第一个参数是文件指针,第二个参数format是格式字符串,用于指定要写入的数据的格式,第三个参数...是要写入的数据,fprintf函数会根据格式字符串的指定,将数据按指定格式写入到文件中,它的返回值为成功写入的字符数量。

我们直接使用实例来加深对其的理解:

#include<stdio.h>
struct Student
{
	char name[20];
	int age;
};
int main()
{
	struct Student stu = { "zhangsan",20};
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//使用fprintf函数格式化将内容写入文件
	fprintf(pf, "%s %d", stu.name, stu.age);
	fclose(pf);
	pf = NULL;
	return 1;
}

这样我们就完成了数据的格式化输入。

        补充:

区分printf 、fprintf 、 sprintf函数  + 区分scanf 、fscanf 、sscanf函数

让我们先来看printf 、fprintf 、 sprintf这一组,先看一下这几个函数的定义:

printf函数:

fprintf函数:

sprintf函数:

根据之前所学习的fprintf函数的几个参数,我们可以知道:

printf:将数据格式化的输出在标准输出流;

fprintf:将数据格式化的输出在任意指定的输出流中;

sprintf:将数据格式化转换为字符串

这里就不在对printf和fprintf函数进行解释了(上文中已解释),只对sprintf函数进行解释:

我们直接使用案例讲解:

#include<stdio.h>
struct Student
{
	char name[10];
	int age;
};
int main()
{
	struct Student stu = { "zhangsan",20 };
	char str[20] = { 0 };
	//将stu结构体中的数据格式化转换为字符串后储存到str数组中
	sprintf(str, "%s %d", stu.name, stu.age);
	printf("%s", str);

	return 0;
}

从结果我们可以看出,sprintf函数将结构体中的数据格式化的转换成了字符串并储存到了str中。

区分完了printf 、fprintf 、 sprintf函数的区别,下面在让我们区分一下scanf 、fscanf 、sscanf这几个函数的区别,先看一下这几个函数的定义:

scanf函数:

fscanf函数:

sscanf函数:

根据之前所学习的fscanf函数的几个参数,我们可以知道:

scanf:将数据格式化的输出在标准输入流;

fscanf:将数据格式化的输出在任意指定的输出流;

sscanf:将字符串按照一定格式转换为格式化数据;

这里就不在对scanf和fscanf函数进行解释了(上文中已解释),只对sscanf函数进行解释:

我们直接使用案例讲解:

#include<stdio.h>
struct Student
{
	char name[10];
	int age;
};
int main()
{
	struct Student stu = { "zhangsan",20 };
	struct Student stu2 = { 0 };
	char str[20] = { 0 };
	//将stu结构体中的数据格式化转换为字符串后储存到str数组中
	sprintf(str, "%s %d", stu.name, stu.age);

	//将str中的数据格式化输出到stu2结构体中
	sscanf(str, "%s %d", stu2.name, &(stu2.age));
	printf("%s %d", stu2.name, stu2.age);
	return 0;
}

这样我们就完成了将字符数组中的字符串格式化输出了。

        在之前我们知道文件可分为文本文件和二进制文件,那么之前我们学习的都是将数据存储到文本文件中,那么如果我们想存储二进制数据,那该怎么办呢?接下来我们就讲解储存二进制数据的方法。

        (3)fwrite函数

先看一下官网对其的解释:

其中的各个参数分别为ptr是一个指向数据存储位置的指针;size是每个数据项的大小;count是要写入的数据项数量;stream是要写入的文件指针,该函数的返回值为实际写入的数据项数量。

现在我们直接使用实例让你更好的理解:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "wb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int arr[5] = { 1,2,3,4,5 };
	//将arr中的数据以二进制的形式写入文件test.txt中
	fwrite(arr, sizeof(int), 5, pf);
	fclose(pf);
	pf = NULL;

	return 0;
}

注:由于是二进制写入,所以我们会看不懂其中的内容。实际上fwrite函数的作用就是将指定位置的数据写入到文件中。

        (4)fread函数

先看一下官网对其的解释:

其中的各个参数分别为ptr是一个指向数据存储位置的指针;size是每个数据项的大小;count是要读取的数据项数量;stream是要读取的文件指针,该函数的返回值为实际读取的数据项数量。

现在我们继续使用上面fread函数实例让你更好的理解:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "rb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int arr[5] = { 0 };
	//将文件test.txt中的二进制的数据读取到arr数组中
	fread(arr, sizeof(int), 5, pf);
	//打印读取到的数据
	for (int i = 0; i < 5; i++)
	{
		printf("%d ", arr[i]);
	}
	fclose(pf);
	pf = NULL;

	return 0;
}

注:由于是二进制读取,所以可以从文件中读取二进制数据,实际上fread函数的作用就是从文件中读取指定数量的数据项,存储到指定位置。

以上就是文件顺序读写的所有方法了!!!

2.文件的随机读写

了解了文件的顺序读写之后,那么我们就产生了一个问题——那么读写到文件的数据能不能不按照顺序呢?其实是可以的。

        这里我们就要讲解文件的随机读写的函数:

        大致为以下三个,分别为fseek、 ftell、 rewind:

        

        (1)fseek函数

fseek函数用于将文件指针定位到指定位置,它的声明如下:

其中的参数分别为stream是要定位的文件指针,offset是相对于origin的偏移量,origin是定位的起始位置,可以是以下值之一:

SEEK_SET:从文件开头开始计算偏移量;

SEEK_CUR:从当前位置开始计算偏移量;

SEEK_END:从文件末尾开始计算偏移量;

该函数的返回值为一个整数,返回0表示定位成功,返回非零值表示定位失败。

我们直接使用实例来讲解:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//将文件指针从开头偏移6个单位
	fseek(pf, 6, SEEK_SET);
	//获取一个字符
	int ch = fgetc(pf);
	printf("%c", ch);
	fclose(pf);
	pf = NULL;

	return 0;
}

所以一句话总结fseek函数的作用就是移动文件指针的位置,使你能更灵活的读写文件。

        (2)ftell函数

ftell函数用于获取文件指针当前的位置。它的声明如下:

该函数的参数stream是要获取位置的文件指针,函数的返回值为当前文件指针的位置到开始位置的偏移量,以字节为单位,并且如果获取位置失败,ftell函数会返回-1。

我们直接使用实例来讲解:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//将文件指针从开头偏移6个单位
	fseek(pf, 6, SEEK_SET);
	//打印偏移量
	int ret = ftell(pf);
	printf("%d", ret);
	fclose(pf);
	pf = NULL;

	return 0;
}

这样我们就了解了ftell函数的使用方式。

        (3)rewind函数

rewind函数用于将文件指针定位到开始位置。它的声明如下:

该函数的参数stream是要获取位置的文件指针,该函数就是将文件指针定位到开始位置,该函数无返回值。

我们直接使用实例来讲解:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//将文件指针从开头偏移6个单位
	fseek(pf, 6, SEEK_SET);
	//打印偏移量
	int ret = ftell(pf);
	printf("%d\n", ret);

	//将文件指针定位到开始位置
	rewind(pf);
	ret = ftell(pf);
	printf("%d", ret);
	fclose(pf);
	pf = NULL;

	return 0;
}

这样我们就了解了rewind函数的使用方式。

3.文件读取结束的判定

我们知道使文件读取结束有两种可能:

        (1)文件在读取的过程中出现异常;

        (2)文件读取到了文件的末尾;

那么我们如何去判断文件读取结束时是因为出现异常而停止的还是因为读取到文件的末尾而停止的呢?

        这里我们要介绍两个函数:feof和ferror

        (1)feof函数

该函数的参数是要判断的文件的文件指针,当是由于读取到文件末尾而停止则返回非0整数,反之为0。

feof 的作用是:当文件读取结束的时候,判断是读取结束的原因是否是:遇到文件尾结束。

        (2)ferror函数

该函数的参数是要判断的文件的文件指针,当是由于读取到文件末尾而停止则返回非0整数,反之为0

ferror的作用是:当文件读取结束的时候,判断是读取结束的原因是否是:发生异常导致读取结束。

这里我们直接使用案例来加深理解:

#include <stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL) {
		perror("fopen");
		return 1;
	}
	//fgetc 当读取失败的时候或者遇到文件结束的时候,都会返回EOF
	int ch = 0;
	while ((ch = fgetc(pf)) != EOF) 
	{
		printf("%c", ch);
	}
	//判断是什么原因结束的
	if (ferror(pf))
		printf("异常导致");
	else if (feof(pf))
		printf("读取结束导致");
	fclose(pf);
	pf = NULL;

	return 0;
}

这样我们就了解了feof和ferror函数的使用方法。

观看文件操作(上)---------------------------------------------------------------------------------------------------->文件操作(上)(想要了解如何操作文件,那么看这一片就足够了!)-CSDN博客


以上就是文件读取的所有内容了~~~

  • 89
    点赞
  • 53
    收藏
    觉得还不错? 一键收藏
  • 34
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 34
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值