C语言的文件操作

什么是文件

简单地说,磁盘上的文件就是文件。文件由两部分构成:文件内容+文件属性。
在程序设计中我们一般谈的文件有两种:程序文件和数据文件。
程序文件
包括如下三类:
源程序文件(后缀为.c)
目标文件(windows环境后缀为.obj)如:Debug里面存在的临时文件。
可执行程序(windows环境后缀为.exe)。存放于硬盘上,是给编译器看的,所以打开会是一些乱码。
数据文件
数据文件顾名思义肯定是存放数据的文件,存放的是程序运行时读写的数据。
本博客剖析的是数据文件。

文件名

文件标识常被称为文件名。
一个文件要有一个唯一的文件标识,便于用户识别和引用,且同一目录下不可能有两个相同的文件名。
文件名由三部分组成:文件路径+文件名主干+文件后缀。
例如:c:\code\test.txt
windows中,“\”是目录分隔符,“:”是盘符。

文件类型

根据数据的组织形式,数据文件有两种类型:二进制文件文本文件
(excel和word存储的都为二进制)
数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件
如果要求在外存上以ASCII码的形式存储,则需要在存储前转换,以ASCII码字符形式存储的文件就是文本文件

那么,一个数据在内存中到底是怎么存储的?
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。
例如有一个整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中会占用5个字节(每个字符占一个字节)。如果以二进制的形式输出,则在磁盘上只占4个字节。
我们可以通过代码来测试一下:

#include<stdio.h>
#include<stdlib.h>

int main()
{
	int a = 10000;
	FILE* pf = fopen("test.txt", "wb");//打开一个文件
	fwrite(&a, sizeof(int), 1, pf);//往文件里写入
	fclose(pf);//关闭文件
	//pf = NULL;
	system("pause");
	return 0;
}

运行完毕后,是否真的生成了这个test.txt文件呢?我们来验证一下。
我们找到解决方案资源管理器,右击刚刚建立的这个项目,点击在文件资源管理器中打开文件夹。
在这里插入图片描述
然后我们可以很清楚的看到,在这个项目路径下面生成了一个test文件。
在这里插入图片描述
然后我们需要查看里面的内容,还是找到解决方案资源管理器,右击源文件,添加现有项,把刚刚生成的test文件添加进来。
在这里插入图片描述
这个时候如果直接双击test文件打开,那么我们一定会看到一堆乱码。右击test文件,选择打开方式,选择二进制编辑器。
在这里插入图片描述
然后我们就可以清楚地看到在test文件里10000是以二进制显示的。
在这里插入图片描述

文件缓冲区

ANSIC标准采用“缓冲文件系统”处理数据文件,所谓缓冲文件系统是指系统自动地在内存中为程序每一个正在使用的文件开辟一块“文件缓冲区”。
从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后再一起送到磁盘上。如果磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区)然后再从缓冲区逐个将数据送到程序数据区。
在这里插入图片描述
缓冲的三种方式:

  • 无缓冲:数据直接到目标区域 Write through(写透)可解决CPU与内部速度不匹配,提高应用程序效率。
  • 行缓冲:显示器,碰到\n将之前的数据输出
  • 全缓冲

文件指针

强调一下,文件指针是缓冲文件系统中较为关键的概念,全称叫文件类型指针
当每个被使用的文件都在内存中开辟了一个文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些变量保存在一个结构体变量中的。这个结构体类型取名叫FILE
不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。每打开一个文件的时候,系统就会根据文件的情况 自动创建一个FILE结构的变量,并填充其中的信息。

所以这个时候就有了FILE的出现。一般都是通过一个FILE的指针来维护这个FILE结构的变量。

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

FILE* PF;

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

文件的打开和关闭

我们要明白,文件在读写之前应该先打开文件,在使用结束后关闭文件
在编写一个程序时,每打开一个文件,都会返回一个 FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系。ANSIC规定,使用fopen函数来打开文件,fclose来关闭文件。

FILE * fopen(const char * filename, const char * mode);
int fclose (FILE * steam);

打开一个文件实例代码:

#include<stdio.h>
int main()
{
	FILE *pFile;
	pFile = fopen("test.txt", "w");
	if (pFile != NULL)
	{
		fputs("fopen example", pFile);
		fclose(pFile);
	}
	return 0;
}

文件的顺序读写

功能函数名
字符输入函数fgetc
字符输出函数fputc
文本行输入函数fgets
文本行输出函数fputs
格式化输入函数fscanf
格式化输出函数fprintf
二进制输入fread
二进制输出fwrite

文件的随机读写

fseek:根据文件指针的位置和偏移量来定位文件指针。

int fseek(FILE * stream, long int offset, int origin);

FILE * stream:想改的文件。
long int offset:偏移量,往前为+,往后为-。
int origin:标志起始位置。一般有SEEK_SET(起始位置)、SEEK_CUR(当前位置)、SEEK_END(最后的位置,这种情况下偏移量为负数)。

我们来看一段代码:

#include<stdio.h>
#include<stdlib.h>
int main()
{
	FILE*pfile;
	pfile = fopen("test.txt","wb");
	fputs("This is an apple.", pfile);//pfile可直接写成stdout,写到显示器输出。
	fseek(pfile, 9, SEEK_SET);
	fputs("sam", pfile);
	fclose(pfile);
	system("pause");
	return 0;
}

这段代码的核心功能就是利了fseek函数找到文件写入的字符串中相对于起始位置的偏移量为9的位置。然后用fputs函数把这个位置之后的三个字符替换成了sam。我们可以打开这个文件来查看一下:
在这里插入图片描述
ftell:返回文件指针相对于起始位置的偏移量.

long int ftell(FILE * stream);

实例代码:

#include<stdio.h>
#include<stdlib.h>
int main()
{
	FILE*pfile;
	long size;
	pfile = fopen("test.txt", "rb");
	if (pfile == NULL)perror("Error opening file");
	else
	{
		fseek(pfile, 0, SEEK_END);
		size = ftell(pfile);
		fclose(pfile);
		printf("Size of test.txt:%ld bytes.\n", size);
	}
	system("pause");
	return 0;
}

这段代码的核心功能是先通过fseek找到相对于字符串结束位置偏移量为0的位置,然后用ftell得到这个数字。
运行结果如下:
在这里插入图片描述
rewind:让文件指针的位置回到文件的起始位置。

void rewind(FILE * steam);

文件结束判定

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

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值