文件的操作

一、首先我们要搞清楚为什么要使用文件呢?
如果没有文件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失 了,等再次运行程序,是看不到上次程序的数据的,如果要将数据进行持久化的保存,我们可以使用文件。
二、什么是文件呢?

简单来说磁盘(硬盘)上的文件是文件。

在程序设计中,我们按照文件的功能可以分为两种
程序文件程序文 件包括源程序文 件(后缀为.c),目 标文 件(windows环境后缀为.obj),可执行 (windows 环境后缀为.exe)。

数据文件文件的内容不⼀定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件,而这次讨论文件的操作就是基于数据文件而进行的。

文件名

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

文件名包含3部分:文件路径+文件名主干+文件后缀

例如: D:\vs\test.txt

为了方便起见,文件标识常被称为文件名。
三、二进制文件和文本文件

根据数据的组织形式,把数据文件分为二进制文件和文本文件。

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

一个数据在文件中是怎样储存的呢?

假如我们有一个数据10000那么他在两种文件中是怎么存储的呢?我们可以看下面这张图

这时候你们可能好奇为什么以ASCII值的形式存储会占五个字节

这是因为如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占⽤5个字节(每个字符⼀个字节),而二进制形式输出,则在磁盘上只占4个字节(VS2019测试)。

:字符一律以ASCII值的形式存储,而整数类型则两种都可以存储,既可以是ASCII值类型也可以是以二进制的形式存储。

我们可以看一段小代码看是否像我们说的那样

int main()
{
	int a = 10000;
	FILE* pf = fopen("text.txt", "wb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fwrite(&a, 4, 1, pf);
	fclose(pf);
	pf = NULL;
	return 0;
}

我们在vs中打开它

我们可以看到他的二进制,但在vs中数据是以十进制存放的我们可以换算一下转换凡是如下

四、文件的打开和关闭
1.流和标准流
1.1、流
我们程序的数据需要输出到各种外部设备,也需要从外部设备获取数据,不同的外部设备的输⼊输出 操作各不相同,为了方便程序员对各种设备进进方便的操作,我们抽象出了流的概念,我们可以把流 想象成流淌着字符的河。
C程序针对文件、画面、键盘等的数据输⼊输出操作都是通过流操作的。
⼀般情况下,我们要想向流里写数据,或者从流中读取数据,都是要打开流,然后操作。
1.2、标准流
那为什么我们从键盘输⼊数据,向屏幕上输出数据,并没有打开流呢?
那是因为C语言程序在启动的时候,默认打开了3个流:
  stdin - 标准输入流,在大多数的环境中从键盘输入,scanf函数就是从标准输入流中读取数据。
  stdout - 标准输出流,大多数的环境中输出至显示器界面,printf函数就是将信息输出到标准输出
流中。
  stderr - 标准错误流,大多数环境中输出到显示器界界。
这是默认打开了这三个流,我们使用scanf、printf等函数就可以直接进进输入输出操作的。
stdin、stdout、stderr 三个流的类型是: FILE* ,通常称为文件指针。
C语言中,就是通过 FILE* 的文件指针来维护流的各种操作的。
1.3、文件指针

缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”。
每个被使同的问件都在内存中开辟了⼀个相应的文件信息区,同来存放文件的相关信息(如文件的名 字,文件状态及文件当前的位置等)。这些信息是保存在⼀个结构体变量中的。该结构体类型是由系 统声明的,取名 FILE.
在stdio.h中的类型有下面这些(在vs2013)中
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结构的变量,这样使用起来更加方便
定义pf是⼀个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区(是⼀个结构体变 量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够间接找到与 它关联的文件。
1.4、文件的读写和关闭

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

打开用fopen

这里的mode指的是文件的打开方式

结束用fclos

更多的详细信息可以到cplusplus上查看更多信息网址如下

https://cplusplus.com/

文件的打开方式很多如下图所示

我们来举几个例子实践一下

int main ()
{
 FILE * pFile;
 //打开⽂件
 pf = fopen ("text.txt","w");
 if(pf==NULL)
 {
 perror("fopen");//如果文件没打开成功打印出错误的原因
 retrun 1;
 }
 //⽂件操作
 if (pf!=NULL)
 {
 fputs ("fopen example",pf);//将内容写入text.txt中
 //关闭⽂件
 fclose (pf);
 }
 return 0;
}

如果没有报错那么打开text.txt会看到,这样就成功的将fopen example存入文件text.txt中了

五、文件的顺序读写

文件的顺序读写函数有很多,且两两相识,很容易记混淆

接下来我们一一了解这几种函数

5.1、fputc

参数

int main()
{
	//打开⽂件
	FILE* pf = fopen("text.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}

	//写文件
	fputc('a', pf);
	fputc('b', pf);
	fputc('c', pf);

	//关闭⽂件
	fclose(pf);
	pf = NULL;
	return 0;
}

5.2、fgets
int main()
{
	//打开⽂件
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int ch = fgetc(pf);
	printf("%c\n", ch);//a

	ch = fgetc(pf);
	printf("%c\n", ch);//b

	ch = fgetc(pf);
	printf("%c\n", ch);//c
	ch = fgetc(pf);
	printf("%c\n", ch);//d
	//关闭⽂件
	fclose(pf);
	pf = NULL;
	return 0;
}

5.3、fputs

int main()
{
	//打开⽂件
	FILE* pf = fopen("text.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("hello word", pf);
	//关闭⽂件
	fclose(pf);
	pf = NULL;
	return 0;
}

5.4、fgets

str:指向复制读取的字符串的char数组的指针。

num:要复制到str中的最大字符数(包括终止空字符)

stream:指向标识输入流的FILE对象的指针。
stdin可以用作从标准输入读取的参数。

int main()
{
	//打开⽂件
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//读文件
	char arr[10] = { 0 };
    fgets(arr,10,pf);

	//关闭⽂件
	fclose(pf);
	pf = NULL;
	return 0;
}

注:从流 中读取字符并将它们作为 C 字符串存储到str中,直到读取 ( num -1) 个字符或到达换行符或到达文件末尾(以先发生者为准)。 换行符使fgets停止读取,但该函数将其视为有效字符并包含在复制到str的字符串中。复制到str 的字符后会自动附加一个终止空字符。 请注意,fgets与gets完全不同:不仅fgets接受参数,还允许指定str的最大大小并在字符串中包含任何结尾换行符。

这个时候我们就不得不讲到所有输入流和所有输出流了

我们可以对比一个组函数

scanf/fscanf/sscanf

printf/fprintf/sprintf

这里的scanf和printf就不讲了我们主要讲后面的两种

5.5、fscanf和fprintf

struct S
{
	char name[20];
	int age;
	float score;
};
int main()
{
	struct S s = { "张三",20,85.5 };
	//想把s中的数据存入文件中
	FILE* pf = fopen("text.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 0;
	}
	//	//写文件--以文本的类型写进去的
	fprintf(pf, "%s %d %f", s.name, s.age, s.score);
	
		//关闭文件
		fclose(pf);
		pf = NULL;
		return 0;
	
}

struct S
{
	char name[20];
	int age;
	float score;
};
int main()
{
	struct S s = { 0 };
	//想从text.txt中读取数据放在s中
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 0;
	}

	//读文件
		fscanf(pf, "%s %d %f", s.name, &(s.age), &(s.score));
     //打印在屏幕前
		printf("%s %d %f", s.name, s.age, s.score);

		//关闭文件
		fclose(pf);
		pf = NULL;
		return 0;
	
}

5.6、sscanf和sprintf

struct S
{
	char name[20];
	int age;
	float score;
};
int main()
{
	char str[200] = { 0 };
	struct S s = { "张三",25,85.5f };
	sprintf(str, "%s %d %f", s.name, s.age, s.score);
	printf("以字符串形式%s\n", str);

	struct S t = {0};
	sscanf(str, "%s %d %f", t.name, &(t.age), &(t.score));
	printf("按照格式打印%s %d %f\n", t.name, t.age, t.score);
	return 0;
}

5.7、fwrite

int main()
{
	
	int arr[5] = { 0 };
	FILE* pf = fopen("text.txt", "rb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//写数据
	int sz = sizeof(arr) / sizeof(arr[0]);
	fwrite(arr, sizeof(arr[0]), sz, pf);//以二进制方式写进去的
	
	
	fclose(pf);
	pf = NULL;
	return 0;
}
5.8、fread

int main()
{
	
	int arr[5] = { 0 };
	FILE* pf = fopen("text.txt", "rb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//读数据
	fread(arr, sizeof(arr[0]), 5, pf);
    int i = 0;
    for (i = 0; i < 5; i++)
    {
	     printf("%d ", arr[i]);
     }
	fclose(pf);
	pf = NULL;
	return 0;
}
六、文件的随机读写
6.1、fseek

int main()
{
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//读文件
	int ch = fgetc(pf);
	printf("%c\n", ch);//a

	ch = fgetc(pf);
	printf("%c\n", ch);//b

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

6.2、ftell
int main()
{
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int ch = fgetc(pf);
	printf("%c\n", ch);//a

	fseek(pf, 0, SEEK_END);
	printf("%d\n", ftell(pf));//11
	fclose(pf);
	pf == NULL;
	return 0;
}
6.3、rewind

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

	fseek(pf, -4, SEEK_END);
	
	ch = fgetc(pf);
	printf("%c\n", ch);//f

	rewind(pf);//回到文件起始位置
	ch = fgetc(pf);
	printf("%c\n", ch);
	fclose(pf);
	pf == NULL;
	return 0;
}

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

我们来看一个例子

讲到这里文件的基本操作就已经结束了,下次再见

  • 17
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值