【C语言】文件操作

0.目录

1.什么是文件及文件类型
2.文件名
3.文件缓冲区
4.文件指针
5.文件的打开与关闭
6.文件的顺序读写
7.文件的随机读写
8.文件结束判定

1.什么是文件及文件类型

文件可以分为程序文件数据文件

1.程序文件,包括源程序文件(.c)、目标文件(windows平台下为.obj)和可执行文件(windows平台下为.exe)
2.数据文件,包括程序运行时所需要从中读取数据的文件,或者需要向其中写入数据的文件。根据数据的组织形式,数据文件又可以分为文本文件二进制文件

我们都知道,数据在内存中以二进制的形式进行存储,如果不加以转换直接输入到外存,这种形式的存储文件就是二进制文件;而如果要求数据在外存中以ASCII码的形式存在,则需要在存储前在内存中进行转换,我们说,以ASCII码的形式存储数据的文件就是文本文件

这就要提到数据在内存中的存储形式了。字符一律以ASCII码的形式存储,而数值型数据既可以是二进制的形式,也可以是ASCII码的形式。

今有整数8848,我们来探究一下8848分别以ASCII和二进制的形式在内存中究竟是怎么存储的,请看图一:
在这里插入图片描述

图一

下面,我们以代码的形式来展示如何将8848以二进制的形式写入到数据文件当中,请看代码一:
//代码一
#include<stdio.h>
int main()
{
	int num = 8848;
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror(fopen);
		pf = NULL;
	}
	fwrite(&num, sizeof(num), 1, pf);
	fclose(pf);
	pf = NULL;
	return 0;
}

运行代码后,我们首先打开对应的test.txt文件,发现是人类无法读懂的机器码,如图二:

在这里插入图片描述

图二

于是,我们用VS2019自带的二进制编辑器打开test.txt文件,如图三、四:

在这里插入图片描述

图三

在这里插入图片描述

图四

图四中的数是小端存储的16进制数,故为
0x00002290

即为十进制

8848

2.文件名

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

文件名包含三个部分:文件路径、文件名主干、文件后缀
例如:

C:\Users\imdan\test.c

为了方便起见,文件标识通常被叫做文件名

3.文件缓冲区

文件缓冲区是用以暂时存放读写期间的文件数据而在内存区预留的一定空间。
使用文件缓冲区可减少读取硬盘的次数。
文件缓冲区是用以暂时存放读写期间的文件数据而在内存区预留的一定空间。
请看图五:
在这里插入图片描述

图五

4.文件指针

在C语言中,每个被使用的文件在内存中都会被开辟一块文件信息区用来存放这个文件的各种信息(名字,状态,位置等),我们用一个结构体变量来保存这些信息:

typedef struct
{
short level;
unsigned flags;
char fd;
unsigned char hold;
short bsize;
unsigned char *buffer;
unsigned ar *curp;
unsigned istemp;
short token;
}FILE;

该结构体被命名为FILE

用一个指针变量指向这个文件信息区,这个指针称为文件指针。通过文件指针就可对它所指的文件进行各种操作。

FILE* fp;

在这里,fp就是一个文件指针变量,我们通过fp指针找到文件信息区,通过文件信息区中的信息访问文件。也就是说,通过fp能找到我们要访问的文件。如图六:
在这里插入图片描述

图六

5.文件的打开与关闭

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

我们来看函数fopen与fclose的参数与返回值:

FILE *fopen( const char *filename, const char *mode );

对于fopen,我们需要传入的参数是文件名(Filename)和 存取文件的类型(Type of access permitted),而返回值是一个FILE*的指针。特别地,如果打开文件失败,会返回NULL

int fclose( FILE *stream );

对于fclose,我们需要传入的参数是FILE*的指针,如果关闭文件成功,返回0;关闭失败,返回EOF

下面,让我们来看一下fopen中的第二个参数mode究竟有哪些(见表一)

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

接下来,让我们举一个小例子。我们想把"helloworld"这个字符串写入test.txt的文件中。请看代码二:
//代码二
#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror(fopen);
		pf = NULL;
	}
	fputs("helloworld", pf);
	fclose(pf);
	pf = NULL;
	return 0;
}

6.文件的顺序读写

下面,我们来介绍一些与文件读写有关的函数(见表二)

功能函数名适用范围定义
字符输入函数fgetc所有输入流int fgetc( FILE *stream );
字符输出函数fputc所有输出流int fputc( int c, FILE *stream );
文本行输入函数fgets所有输入流char *fgets( char *string, int n, FILE *stream );
文本行输出函数fputs所有输出流int fputs( const char *string, FILE *stream );
格式化输入函数fscanf所有输入流int fscanf( FILE *stream, const char *format [, argument ]… );
格式化输出函数fprintf所有输出流int fprintf( FILE *stream, const char *format [, argument ]…);
从字符串读取格式化输入sscanf字符串int sscanf( const char *buffer, const char *format [, argument ] … );
将格式化输出写入字符串sprintf字符串int sprintf( char *buffer, const char *format [, argument] … );
二进制输入fread文件size_t fread( void *buffer, size_t size, size_t count, FILE *stream );
二进制输出fwrite文件size_t fwrite( const void *buffer, size_t size, size_t count, FILE *stream );
表二

7.文件的随机读写

下面,我们来继续介绍几个有关文件随机读写的函数,fseekftell以及rewind

7.1 fseek

将文件指针移动到指定位置

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

stream表示流,是一个FILE*的指针
offset表示偏移量(单位是字节),1代表向后偏移一个字节,-1代表向前偏移一个字节
origin表示起始位置,有三种:SEEK_CUR(当前文件指针指向的位置) SEEK_END(文件结尾) SEEK_SET(文件开头)

7.2 ftell

返回当前文件指针的位置,即相对于文件开始时的偏移量

long ftell( FILE *stream );

7.3 rewind

将文件指针重置到文件开始位置

请看代码三的演示,我们将fseek、ftell、rewind综合使用:

//代码三
#include<stdio.h>
int main()
{
	char arr[30] = { 0 };
	//以读写的方式建立新文件
	FILE* fp = fopen("test.txt", "w+");
	if (fp == NULL)
	{
		perror(fopen);
		return 1;
	}
	//向文件中写入字符串
	fputs("It is an apple.\n", fp);
	//重置fp位置,输出文件内容至屏幕
	rewind(fp);
	fgets(arr, 18, fp);
	fprintf(stdout,"%s", arr);
	//将fp的位置移动至开头位置后8个字节处,并使用ftell查看此时的偏移量
	fseek(fp, 7, SEEK_SET);
	printf("offset = %d\n", ftell(fp));
	//在偏移量为8的位置写入字符串,此时覆盖了原本的由此向后的内容
	fputs(" banana.", fp);
	//重置fp位置,输出文件内容至屏幕
	rewind(fp);
	fgets(arr, 25, fp);
	fprintf(stdout, "%s", arr);
	//关闭文件并将fp置为空指针
	fclose(fp);
	fp = NULL;
	return 0;
}

运行结果见图七:
在这里插入图片描述

图七

8.文件结束判定

int feof( FILE *stream );

在文件读取结束的时候,我们使用feof函数来判断文件是因为读取失败结束还是遇到文件尾结束
关于feof的返回值,如果文件正常结束即文件指针指向文件尾,返回非零值;如果结束时文件指针没有指向文件结尾,返回0
需要强调的是,feof函数只能用来判断文件是以什么方式结束,而不能用来判断文件是否结束

我们能够明白为什么feof不能用来判断文件是否结束,若我们错误地以feof的返回值来判断文件是否结束,若返回值为0,你以为是文件还没有结束,但是若当文件出错结束时,结果就和文件未结束一样,都会返回0

所以,我们要用其他方式首先判断文件是否已经结束,然后才能用feof来判断文件究竟以什么方式结束的

8.1 文本文件

文本文件读取是否结束,判断返回值为EOF(fgetc),或者NULL(fgets)

在文件已经结束时,fgetc判断以何种方式结束的方法是判断返回值是否为EOF(EOF to indicate an error or end of file)
在文件已经结束时,fgets判断以何种方式结束的方法是是判断返回值否为NULL(NULL is returned to indicate an error or an end-of-file condition.)

请看代码四:

我们事先创建一个test.txt文件,并在其中写入helloworld的内容

//代码四
#include<stdio.h>
int main()
{
	FILE* fp = fopen("test.txt", "r+");
	if (!fp)
	{
		perror(fopen);
		return 1;
	}
	//读取test.txt中的字符,遇到文件尾则停止
	char c = 0;
	while ((c = fgetc(fp))!=EOF)
	{
		putchar(c);
	}
	//判断是否是读取失败结束
	if (ferror(fp))
	{
		puts("\nI/O error when reading\n");
	}
	//判断是否是遇到文件尾结束
	else if (feof(fp))
	{
		puts("\nEnd of file reached successfully\n");
	}
	fclose(fp);
	fp = NULL;
	return 0;
}

运行结果
在这里插入图片描述

图八

8.2二进制文件

二进制文件的读取结束的判断为返回值是否小于实际要读的个数,请看代码五:

#include<stdio.h>
int main()
{
	size_t ret = 0;
	int tmp = 0;
	int arr[] = { 1,2,3,4,5 };
	FILE* fp = fopen("test.txt", "wb");//以二进制写的形式打开文件

	if (!fp)
	{
		perror(fopen);
		return 1;
	}
	fwrite(arr, sizeof(arr), 1, fp);//以二进制形式向文件中写入arr数组
	fclose(fp);
	fp = fopen("test.txt", "rb");//以二进制读的形式打开文件
	while((ret = fread(&tmp, sizeof(int), 1, fp)) == 1)//每次读取4个字节的数据
	{
		printf("%d ", tmp);
	}
	//判断是否是读取失败结束
	if (ferror(fp))
	{
		puts("\nI/O error when reading\n");
	}
	//判断是否是遇到文件尾结束
	else if (feof(fp))
	{
		puts("\nEnd of file reached successfully\n");
	}
	fclose(fp);
	fp = NULL;
	return 0;
}

运行结果见图九:
在这里插入图片描述

图九

  • 12
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 19
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

DanteIoVeYou

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值