文件操作(C语言)

目录

一、为什么使用文件

二、什么是文件

2.1程序文件

2.2数据文件

2.3文件名

三、文件的打开和关闭

3.1文件指针

3.2文件的打开和关闭

   四、文件的顺序读写

4.1对比一组函数:

五、文件的随机读写

5.1fseek

5.2ftell

5.3rewind

六.文本文件和二进制文件

七.文件读取结束的判定

八.文件缓冲区


一、为什么使用文件

        像之前写的代码运行出来的程序,保存的数据都会随着程序的退出而消失,这个时候使用文件我们就可以将数据直接存放在电脑硬盘上,做到数据的持有话。

二、什么是文件

        磁盘上的文件是文件。

        但在我们程序设计这种,我们一般谈的文件有两种:程序文件、数据文件(从文件功能角度分类)。

2.1程序文件

        包括源程序文件(后缀为.c),目标文件(Windows环境后缀为.obj),可执行程序(windows环境后缀为.exe)

2.2数据文件

        文件内容不一定是程序,而程序运行时读写的程序,比如程序运行需要从中读取数据的文件或者输出内容的文件。

        本文论述的是数据文件。

        在以前个章所处理数据的输入输出都是以终端为对象的,即从终端的键盘输入数据,运行结果显示到显示器上。其实有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使用,这里处理的就是磁盘上的文件。

2.3文件名

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

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

例如:c:\code\test.txt

为了方便起见,文件标识常被称为文件名

三、文件的打开和关闭

3.1文件指针

        缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”。

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

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

        FILE* pf//文件指针变量

3.2文件的打开和关闭

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

//打开文件
FILE* fopen(const char* filename, const char* mode );

//关闭文件
int fclose(FILE* stream);

        打开方式如下:

        

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

实例代码:

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

   四、文件的顺序读写

       

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

4.1对比一组函数:

scanf/fscanf/sscanf

printf/fprintf/sprintf

scanf--格式化的输入函数

printf--格式化的输出函数

fscanf--针对所有输入流的格式化的输入函数

fprintf--针对所有输出流的格式化的输出函数

sscanf--把一个字符串转化成格式化的数据

sprintf--把一个格式化的数据转化成字符串

文件流:

//只要一个C程序运行起来,这3个流就是默认打开的

标准输入流:stdin   //键盘

标准输出流:stdout //屏幕

标准错误流:stderr //屏幕

实例代码:

#include<stdio.h>

struct S
{
	int a;
	float s;
	char str[10];
};
int main()
{
	char arr[30] = { 0 };
	struct S s = { 100,3.14f,"haha" };
	struct S tmp = { 0 };
	sprintf(arr, "%d %f %s", s.a, s.s, s.str);//把格式化的数据写入字符串
	/*printf("%s\n", arr);*/
	sscanf(arr, "%d %f %s", &(tmp.a), &(tmp.s), &(tmp.str)); //从字符串中读取格式化数据
	printf("%d %f %s\n", tmp.a, tmp.s, tmp.str);
	return 0;
}

五、文件的随机读写

5.1fseek

//根据文件指针的位置和偏移量来定位文件指针

int fseek(FILE* stream, long int offset, int origin);//文件指针,偏移量,起源

起源:

SEEK_CUR    : 文件指针当前位置

SEEK_END    :文件末尾的位置

SEEK_SET    :文件开始的位置

实例代码:

int main()
{
	FILE* pf = fopen("test.txt", "wb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("hello world", pf);
	fseek(pf, 5, SEEK_SET);
	fputs("+", pf);//直接替换掉中间空格
	fclose(pf);
	return;
}
int main()
{
	FILE* pf = fopen("test.txt", "rb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int arr[30] = { 0 };
	fgets(arr, 12, pf);//12是读取个数
	printf("%s", arr);
	return 0;
}

当使用SEEK_END时偏移量为负数。

5.2ftell

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

long int ftell(FILE* stream);

5.3rewind

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

void rewind(FILE* stream);

六.文本文件和二进制文件

根据数据的组织形式,数据文件被称为文本文件或者二级制文件。

数据在内存中以二进制的形式存储,如果不加转换的输出到外存就是二进制文件。

如果要求在外存上以ASCII码的形式存储 ,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件。

一个数据在内存中怎样存储的呢?

字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。

如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占用5个字节(每个字符一个字节),而二进制形式输出,则在磁盘上只有4个字节(VS2022测试)

内存中存储形式:00000000 00000000 00100111 00010000 

ASCII形式         :00110001 00110000 00110000 00110000 00110000

二进制形式        : 00000000 00000000 00100111 00010000 

测试代码:

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

用二进制编辑器打开test.txt:

七.文件读取结束的判定

在文件读取中,不能直接用feof函数的返回值来判断文件的是否结束

而是应用于文件读取结束的时候,判断是读取失败结束,还是遇到文件末尾结束

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

        fgetc判断是否为EOF。

        fgets判断返回值是否为NULL。

二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。

 例如:

        fread判断返回值是否小于实际要读的个数。

文本文件的例子:

        

#include<stdio.h>
#include<stdlib.h>
int main()
{
	int c = 0;
	FILE* pf = fopen("test.txt", "r");
	if (!pf)
	{
		perror("fopen");
		return 1;
	}
	while (c = fgetc(pf) != EOF)//标准C I/O读取文件循环
	{
		putchar(c);
	}
	printf("\n");
	if (ferror(pf))//判断是否是发生错误读取结束 
	{
		puts("I/O error when reading");
	}
	else if (feof(pf))//在设置了与流相关联的文件结束指示符的情况下,返回非零值。否则,返回零。
	{
		puts("End of file reached successfully");
	}
	return 0;
}

二进制文件的例子:

enum {
	SIZE = 5
};
int main()
{
	double a[SIZE] = { 1.,2.,3.,4.,5. };
	FILE* pf = fopen("test.bin", "wb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fwrite(a, sizeof * a, SIZE, pf);
	fclose(pf);

	double b[SIZE] = { 0 };
	pf = fopen("test.bin", "rb");
	size_t ret_code = fread(b, sizeof * b, SIZE, pf);
	if (ret_code == SIZE)
	{
		puts("Array read successfully,contents:");
		int n = 0;
		for (n = 0; n < SIZE; ++n)
		{
			printf("%f", b[n]);
			putchar('\n');
		}

	}
	else
	{
		if (feof(pf))
		{
			printf("Error reading test.bin: unsxpecteed end of flie \n");
		}
		else if (ferror(pf))
		{
			perror("Error reding test.bin");
		}
	}
	fclose(pf);
	return 0;
}

八.文件缓冲区

ANSIC 标准采用 缓冲文件系统 处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块“ 文件缓冲区 。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。
  • 23
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值