文件操作详细解析

目录

1. 为什么使用文件

2. 什么是文件

3. 文件的打开和关闭

3.1 文件指针

3.2 文件的打开和关闭 

4. 文件的顺序读写

4.1 fgetc && fputc

4.2 fgets && fputs 

4.3 fscanf && fread

4.4 fread && fwrite

 4.5 对比一组函数:

5. 文件的随机读写

5.1 fseek

5.2 ftell && rewind


1. 为什么使用文件

 当我们写程序并运行时,会把一些信息存到内存中,可一旦退出程序,信息就被销毁,下次运行程序时还要重新输入,这就很难受了!

我们想要只有我们自己选择删除数据的时候,数据才不复存在。

这就涉及到了数据持久化的问题,我们一般数据持久化的方法有,把数据存放在磁盘文件、存放到数据库等方式。 

使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。
 

2. 什么是文件

磁盘上的文件是文件。

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;

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

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

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

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

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

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

3.2 文件的打开和关闭 

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

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

打开方式如下:

文件使用方式含义如果指定文件不存在
“r”(只读)为了输入数据,打开一个已经存在的文本文件出错
“w”(只写)为了输出数据,打开一个文本文件建立一个新的文件
“a”(追加)向文本文件尾添加数据建立一个新的文件
“rb”(只读)为了输入数据,打开一个二进制文件出错
“wb”(只写)为了输出数据,打开一个二进制文件建立一个新的文件
“ab”(追加)向一个二进制文件尾添加数据出错
“r+”(读写)为了读和写,打开一个文本文件出错
“w+”(读写)为了读和写,建议一个新的文件建立一个新的文件
“a+”(读写)打开一个文件,在文件尾进行读写建立一个新的文件
“rb+”(读写)为了读和写打开一个二进制文件出错
“wb+”(读写)为了读和写,新建一个新的二进制文件建立一个新的文件
“ab+”(读写)打开一个二进制文件,在文件尾进行读和写建立一个新的文件

举例:

#include <stdio.h>
int main ()
{
FILE * pFile;
//打开文件
pFile = fopen ("myfile.txt","w");
if (pFile == NULL)//检查文件打开是否有问题
{
    perror("fopen");
    return 1;
}
//文件操作


//关闭文件
fclose(pFile);
pFlie = NULL;

return 0;
}


4. 文件的顺序读写

功能函数名适用于
字符输入函数fgetc所有输入流
字符输出函数fputc所有输出流
文本行输入函数fgets所有输入流
文本行输出函数fputs所有输出流
格式化输入函数fscanf所有输入流
格式化输出函数fprintf所有输出流
二进制输入fread文件
二进制输出fwrite文件

挨个举例:

4.1 fgetc && fputc

4.2 fgets && fputs 

4.3 fscanf && fread

4.4 fread && fwrite

 4.5 对比一组函数:

sscnaf && sprintf 举例:

int main()
{
	char buf[256] = { 0 };
	struct S s = { "zhangsan", 20, 95.5 };
	struct S tmp = { 0 };

	sprintf(buf, "%s %d %lf", s.name, s.age, s.d);

	printf("%s\n", buf);//字符串

	//从buf字符串中提取结构体数据
	sscanf(buf, "%s %d %lf", tmp.name, &(tmp.age), &(tmp.d));
	printf("%s %d %lf", tmp.name, tmp.age, tmp.d);//格式化的形式

	return 0;
}

5. 文件的随机读写

       前面讲到fgetc函数,如果多次从文件得到字符,字符会按照顺序被取出,但是实际中我们也许不想按顺序得到字符,而是按照需求去取得,这样,就不得不介绍文件随机读写涉及的函数。

5.1 fseek



举个例子:

#include <stdio.h>
int main()//实现创建一个test.txt并在里面放入abcdef(不创建"r"读取出出错)
{
	//打开文件
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//随机读
	int ch = fgetc(pf);//a
	printf("%c\n", ch);

	ch = fgetc(pf);//b   因为每读取一个字符后自动跳下一个,所以读完'a' 是 'b'
	printf("%c\n", ch);

	fseek(pf, 4, SEEK_SET);//起始位置偏移4个字符,指向e

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

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

	return 0;
}



5.2 ftell && rewind

ftell :返回文件指针相对于起始位置的偏移量

rewind: 让文件指针的位置回到文件的起始位置

long int ftell ( FILE * stream );
void rewind ( FILE * stream );

举个例子:

int main()
{
	//打开文件
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
        perror("fopen");
		return 1;
	}
	//随机读
	fputc('a', pf);
	fputc('b', pf);
	fputc('c', pf);
	fputc('d', pf);

    fseek(pf, -3, SEEK_CUR);
	fputc('w', pf);//此时文件内容为awcd

	long pos = ftell(pf);//此时相对于起始位置偏移量为2
	printf("%ld\n", pos);

	rewind(pf);//回到起始位置

	pos = ftell(pf);//此时偏移量为0
	printf("%ld\n", pos);//0

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

	return 0;
}



 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值