【hello C语言】文件操作

目录

1. 什么是文件?

2. 程序文件

3. 数据文件

 4. 文件名

5. 文件类型

5.1 二进制文件

5.2 文本文件

5.3 数据在内存中的存储

 6. 文件缓冲区

 7. 文件指针

 8. 文件的打开和关闭

 9. 文件的顺序读写

10. 文件的随机读写

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

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

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

11. 文件结束判定

11.1 feof



【hello C语言】文件操作_第1张图片

 C语言

1. 什么是文件?

在以前C语言的学习中,我们处理数据的输入输出都是以终端为对象的,即从终端的键盘输入数据,这时候数据是存储在内存中的,重新启动程序数据便会消失。运行结果是显示到显示器上,同时也没有持久化的保存数据。

文件是用来持久化保存数据的。一般存储在磁盘上。

在程序设计中,我们一般谈的文件有两种:程序文件、数据文件

2. 程序文件

源文件:后缀为.c

目标文件:windows环境后缀为.obj

可执行程序:windows环境后缀为.exe

3. 数据文件

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

 4. 文件名

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

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

例如:

F:\Users\LENOVO\Desktop\毕设\开题报告

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

5. 文件类型

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

5.1 二进制文件

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

5.2 文本文件

如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。

以ASCII字符的形式存储的文件就是文本文件

5.3 数据在内存中的存储

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

例如:

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

使用如下代码可以通过文件方式查看内存中的二进制显示:

#define _CRT_SECURE_NO_WARNINGS 1
#include 

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

【hello C语言】文件操作_第2张图片

【hello C语言】文件操作_第3张图片

 6. 文件缓冲区

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

【hello C语言】文件操作_第4张图片

 7. 文件指针

缓冲文件系统中,关键的概念是 文件类型指针 ,简称 文件指针
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。
这些信息是保存在一个结构体变量中的。该结构体类型是由系统声明的,取名FILE .
不同的C编译器的 FILE 类型包含的内容不完全相同,但是大同小异。
每当打开一个文件的时候,系统会根据文件的情况自动创建一个  FILE  结构的变量,并填充其中的信息,使用者不必关心细节。
一般都是通过一个 FILE 的指针来维护这个 FILE 结构的变量,这样使用起来更加方便。
下面我们可以创建一个FILE*的指针变量 :
FILE* pf;//文件指针变量
定义 pf 是一个指向  FILE  类型数据的指针变量。可以使 pf 指向某个文件的文件信息区(是一个结构体变量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够找到与它关联的文件
比如:

【hello C语言】文件操作_第5张图片

 8. 文件的打开和关闭

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

在编写程序的时候,在打开文件的同时,都会返回一个 FILE* 的指针变量指向该文件,也相当于建立了指针和文件的关系。
ANSIC 规定使用fopen函数来打开文件,fclose来关闭文件。
FILE* fopen(const char* filename, const char* mode);
int fclose(FILE* stream);

打开的方式如下:

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

示例代码:

#define _CRT_SECURE_NO_WARNINGS 1
#include 

int main()
{
	FILE* pFile;
	pFile = fopen("myfile.txt", "w");
	if (pFile == NULL)
		perror("fopen");
	fputs("fopen example", pFile);
	fclose(pFile);
	pFile = NULL;
	return 0;
}

【hello C语言】文件操作_第6张图片

 9. 文件的顺序读写

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

为了方便更好的使用,在此对比两组函数:
  • scanf 与 fscanf
【hello C语言】文件操作_第7张图片

 【hello C语言】文件操作_第8张图片

  •  printf 与 fprintf
【hello C语言】文件操作_第9张图片

 【hello C语言】文件操作_第10张图片

10. 文件的随机读写

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

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

看个例子:

#define _CRT_SECURE_NO_WARNINGS 1
#include 

int main()
{
	FILE* pFile;
	pFile = fopen("example.txt","wb");
	fputs("This is an apple.", pFile);
	fseek(pFile, 9, SEEK_SET);
	fputs(" sam", pFile);
	fclose(pFile);
	pFile = NULL;
	return 0;
}
【hello C语言】文件操作_第11张图片

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

long int ftell ( FILE * stream );

看个例子:

#define _CRT_SECURE_NO_WARNINGS 1
#include 

int main()
{
	FILE* pFile;
	long size;
	pFile = fopen("myfile.txt", "rb");
	if (pFile == NULL)
		perror("fopen");
	fseek(pFile, 6, SEEK_SET);
	size = ftell(pFile);
	fclose(pFile);
	pFile = NULL;
	printf("Size of myfile.txt:%ld bytes.\n", size);
	return 0;
}

【hello C语言】文件操作_第12张图片

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

void rewind ( FILE * stream );
看个例子:
#define _CRT_SECURE_NO_WARNINGS 1
#include 

int main()
{
	int n;
	FILE* pFile;
	char buffer[27];

	pFile = fopen("myfile.txt", "w+");
	for (n = 'a'; n <= 'z';n++)
	{
		fputc(n, pFile);
	}
	rewind(pFile);
	fread(buffer, 1, 26, pFile);
	fclose(pFile);
	pFile = NULL;
	buffer[26] = '\0';
	puts(buffer);
	return 0;
}

【hello C语言】文件操作_第13张图片

11. 文件结束判定

11.1 feof

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

文本文件的例子:

判断是读取失败结束,还是文件结束的时候结束

#define _CRT_SECURE_NO_WARNINGS 1
#include 
#include 

int main()
{
	int c;
	FILE* fp = fopen("test.txt", "r");
	if (fp == NULL)
		perror("fopen");
	//fgetc当读取失败的时候或者遇到文件结束的时候,都会返回EOF
	while ((c = fgetc(fp)) != EOF)
		putchar(c);
	if (ferror(fp))
		puts("I/O error when reading");
	else
		puts("End of file reached successfully");
	fclose(fp);
	fp = NULL;
	return 0;
}

【hello C语言】文件操作_第14张图片

二进制文件的例子:

#define _CRT_SECURE_NO_WARNINGS 1
#include 
#include 

enum
{
	SIZE = 5,
};

int main()
{
	double a[SIZE] = { 1.0,2.0,3.0,4.0,5.0 };
	double b = 0.0;
	size_t ret_cobe = 0;
	FILE* fp = fopen("test.bin", "wb");
	fwrite(a, sizeof(double), SIZE, fp);
	fclose(fp);
	fp = NULL;

	fp = fopen("test.bin", "rb");
	while ((ret_cobe = fread(&b, sizeof(double), 1, fp)) >= 1)
		printf("%lf\n", b);
	if (feof(fp))
		printf("Error reading test.bin:unexpected end of file\n");
	else if (ferror(fp))
		perror("Error reading tes.bin");
	fclose(fp);
	fp = NULL;
	return 0;
}

【hello C语言】文件操作_第15张图片

坚持打卡!

你可能感兴趣的:(C语言,c语言)