学习C语言之 了解文件操作

了解文件操作

1、为什么使用文件

        之前编写的程序,一旦程序退出,其中所有的数据都会被销毁。想要保存数据,就需要数据的持久化,数据持久化的方法有:把数据存放在磁盘文件、存放到数据库等方式。

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

2、什么是文件

        日常中说的文件是磁盘上的文件。

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

        2.1、程序文件

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

        2.2、数据文件

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

        本篇讨论的是数据文件。

        在以前各篇所处理数据的输入输出都是以终端为对象的,即从终端的键盘输入数据,运行结果显示到显示器上。

        其实有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使用,这里处理的就是磁盘上的文件。

2.3、文件名

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

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

        例如:c:\code\test.txt

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

        C语言编译器会在默认文件路径读取和创建文件,一般是在项目文件夹下。

3、文件的打开和关闭

        3.1、文件指针

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

        每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是由系统声明的,取名 FILE。

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

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

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

        FILE* 指针变量:

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

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

演示如下:

//默认文件夹下打开和关闭文件
//fopen fclose 函数使用
int main()
{
	//VS2019 的默认文件路径是在解决方案的文件夹下
	//如项目名是 test,解决方案名叫solution
	//那么在项目文件夹test里面会有一个文件夹叫solution
	//新创建的文件就会在solution文件夹下
	//读取文件也会在solution文件夹下读取

	FILE* pf = fopen("test.txt", "w");	
	//打开solution文件夹下的test.txt文件,用于向其输出数据。
	//如果文件夹下没有test.txt文件,则新建一个test.txt文件。
	//注:文件名要在 "" 内,w/r/a 等也要在 "" 内。

	if (pf == NULL)	//由于fopen返回值是指针,也要进行空指针判断
	{
		perror("fopen");	//打印错误信息。perror的参数是 "可能出错的函数名"
		return 1;
	}

	fclose(pf);	//关闭文件指针pf指向的文件,这里是test.txt。
	pf = NULL;	//将pf指针置为NULL,防止后续误引用。

	return 0;
}

3.2、文件打开和关闭

        文件在读写之前应该先打开文件,在使用结束之后应该关闭文件。(打开的文件数量是有限的,不关闭会导致后续无法打开新文件)

        在编写程序的时候,在打开文件的同时,都会返回一个FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系。

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

        //打开文件

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

        //关闭文件

        int fclose ( FILE* stream );

演示如下:

//指定文件夹下打开和关闭文件
int main()
{
	
	//直接输入文件路径,也可以找到文件
	//FILE* pf = fopen("C:\Users\Administrator\Desktop\test.txt", "w");	//会报错
	 
	//因为"\" 是转义字符,所以上述输入文件路径会被误判
	//之前提到,"\\"可以表示'\'

	FILE* pf = fopen("C:\\Users\\Administrator\\Desktop\\test.txt", "w");
	//打开桌面文件夹下的test.txt文件,用于向其输出数据。
	//如果文件夹下没有test.txt文件,则新建一个test.txt文件。

	if (pf == NULL)	//由于fopen返回值是指针,也要进行空指针判断
	{
		perror("fopen");	//打印错误信息。perror的参数是 "可能出错的函数名"
		return 1;
	}

	fclose(pf);	//关闭文件指针pf指向的文件,这里是test.txt。
	pf = NULL;	//将pf指针置为NULL,防止后续误引用。

	return 0;
}

        打开方式如下:

        文件使用方式                         含义                                                 如果指定文件不存在

        ”r“(只读)             为了输入数据,打开一个已经存在的文本文件         出错

        “w”(只写)            为了输出数据,打开一个文本文件                           建立一个新的文件

        “a”(追加)             向文本文件尾添加数据                                             建立一个新的文件

        “rb”(只读)            为了输入数据,打开一个二进制文件                       出错

        “wb”(只写)           为了输出数据,打开一个二进制文件                      建立一个新的文件

        “ab”(追加)            向一个二进制文件尾添加数据                                出错

        ”r+“(读写)            为了读和写,打开一个文本文件                             出错

        “w+”(读写)           为了读和写,建立一个新的文件                             建立一个新的文件

        “a+”(读写)            打开一个文件,在文件尾进行读写                         建立一个新的文件

        “rb+”(读写)           为了读和写,打开一个二进制文件                         出错

        “wb+”(读写)          为了读和写,新建一个二进制文件                        建立一个新的文件

        “ab+”(读写)           打开一个二进制文件,在文件尾进行读和写         建立一个新的文件

4、文件的顺序读写

        功能                                         函数名                                 适用于

字符输入函数                                 fgetc                                 所有输入流

字符输出函数                                 fputc                                 所有输出流

文本行输入函数                             fgets                                 所有输入流

文本行输出函数                             fputs                                 所有输出流

格式化输入函数                             fscanf                                 所有输入流

格式化输出函数                            fprintf                                 所有输出流

二进制输入                                    fread                                   文件

二进制输出                                    fwrite                                  文件

流的介绍:

//流的概念
//所谓流,是一个抽象概念。用来表示程序与运行环境(系统)之间进行互动的通道。
//这种互动包括读取和写入数据。

//流的作用
//程序输出数据,可能会保存到硬盘、光盘、U盘等各种外设上,但是每种载体的输入输出规则不通过,
//程序员不可能每次输出数据都去研究怎么输出数据到外设,这样要求太高。
//于是就有了流的概念,即数据输出后以流的形式存在,各种外设接收流之后自行处理并保存。

//可以理解为以前送货,用卡车送油到各个工厂,需要和工厂对接,
//要负责处理食用的、工业用的、粘度大的小的、挥发快的慢的、怎么使用、保存等,太麻烦了。
//有了流,只要将油送到公路上,汇入车流,工厂自己会看是不是自家的油,并自行处理。
//这里工厂就相当于外设,车流就是我们输出的信息流。

//C语言默认会打开三个流
//1、stdin:标准输入流——键盘,
//2、stdout:标准输出流——屏幕,
//3、stderr:标准错误流——屏幕,
//即打开编译器,编译器就会索引信息流,判断是不是键盘输入信息,并接收处理
//编译器还会将信息输出到屏幕上,让我们可以看到程序代码
//编译器一直运行判断,出现错误时,它会输出信息到屏幕,显示错误。
//这三种标准流都是FILE* 类型的。


//直接向屏幕输出字符
//stdout 函数演示
int main()
{
	//标准流的类型是FILE*,可以和文件一样操作
	fputc('h', stdout);	//向屏幕输出字符 h 。
	fputc('e', stdout);	//向屏幕输出字符 e 。
	fputc('l', stdout);	//向屏幕输出字符 l 。
	fputc('l', stdout);	//向屏幕输出字符 l 。
	fputc('o', stdout);	//向屏幕输出字符 o 。

	//运行程序,屏幕出现 hello 。
	//未使用打印程序,仍然输出字符 。


	return 0;
}

文件顺序读写演示如下:

fputc 函数:

//文件的顺序读写 fgetc fputc fgets fputs fcanf fprintf fread fwrite
//向文件写入字符
//fputc 函数使用
int main()
{
	FILE* pf = fopen("test.txt", "w");	//打开默认文件夹下的文件test.txt ,用于从程序向其输出数据。
	//文件夹下没有test.txt时,新建一个test.txt文件。
	//"w"的输出方式会将原文件中的数据清除,重新开始输出。

	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}

	fputc('h', pf);	//向文件指针pf指向的文件输出字符 'h' 。
	//输出字符后,pf指针会自行调整
	//此时使用fputc会在后面继续输出数据
	fputc('e', pf);	//向文件指针pf指向的文件输出字符 'e' 。
	fputc('l', pf);	//向文件指针pf指向的文件输出字符 'l' 。
	fputc('l', pf);	//向文件指针pf指向的文件输出字符 'l' 。
	fputc('o', pf);	//向文件指针pf指向的文件输出字符 'o' 。
	//打开文件test.txt,可以看到,其中已经有了 hello字符串数据。
	//说明fputc是直接在后面继续添加数据。

	
	fclose(pf);	//关闭文件指针pf指向的文件,这里是test.txt。
	pf = NULL;	//将pf指针置为NULL,防止后续误引用。

	return 0;
}

fgetc函数:

//从文件读取字符
//fgetc 函数使用
int main()
{
	FILE* pf = fopen("test.txt", "r");	//打开默认文件夹下的文件test.txt ,用于从其向程序输入数据。
	//文件夹下没有test.txt时,会报错。

	if (pf == NULL)	
	{
		perror("fopen");
		return 1;
	}

	int ch = fgetc(pf);	//从文件指针pf指向的文件读取一个字符。
	//读取字符后,pf指针会自行调整
	//此时使用fputc会在后面继续读取数据
	
	//fgetc 返回的是字符的ASCII码。

	printf("%c ", ch);	//打印结果 h 。

	ch = fgetc(pf);	//从文件指针pf指向的文件读取一个字符。
	printf("%c ", ch);	//打印结果 e 。
	ch = fgetc(pf);	//从文件指针pf指向的文件读取一个字符。
	printf("%c ", ch);	//打印结果 l 。
	ch = fgetc(pf);	//从文件指针pf指向的文件读取一个字符。
	printf("%c ", ch);	//打印结果 l 。
	ch = fgetc(pf);	//从文件指针pf指向的文件读取一个字符。
	printf("%c \n", ch);	//打印结果 o 。
	ch = fgetc(pf);	//从文件指针pf指向的文件读取一个字符。
	printf("%c \n", ch);	//打印结果 '\n' 。
	//进入内存可以看到字符'0'后面是 -1,即文件结束符
	//fgetc返回-1即EOF,文件读取完毕。后续不会再读取出数据

	//说明fputc是直接在后面继续添加数据。


	fclose(pf);	//关闭文件指针pf指向的文件,这里是test.txt。
	pf = NULL;	//将pf指针置为NULL,防止后续误引用。

	return 0;
}

fputs函数:

//向文件写入字符串
//fputs 函数使用
int main()
{
	FILE* pf = fopen("test.txt", "w");	//向文件输出数据
	if (pf == NULL)	
	{
		perror("fopen");
		return 1;
	}

	fputs("abcdef", pf);	//打开文件,可以看到其中有字符串abcdef
	fclose(pf);	//关闭文件
	pf = NULL;
	return 0;
}

fputs函数2:

//向文件写入字符串2
int main()
{
	FILE* pf = fopen("test.txt", "w");	//向文件输出数据
	if (pf == NULL)	
	{
		perror("fopen");
		return 1;
	}

	fputs("abcdef", pf);	//打开文件,可以看到其中有字符串abcdef
	fputs("zxcvb", pf);	//打开文件,可以看到其中有字符串abcdefzxcvb
	//可见fputs不会在字符串后面自动添加 '\0',需要自己添加

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

fgets函数:

//从文件读取字符串
//fgets 函数使用
int main()
{
	FILE* pf = fopen("test.txt", "w");	//向文件输出数据
	if (pf == NULL)	//由于fopen返回值是指针,也要进行空指针判断
	{
		perror("fopen");
		return 1;
	}

	fputs("abcdefghijklmn", pf);	//打开文件,可以看到其中有字符串abcdef
	fclose(pf);	//关闭文件
	pf = NULL;

	FILE* pff = fopen("test.txt", "r");	//从文件输入数据
	if (pff == NULL)	
	{
		perror("fopen");
		return 1;
	}

	char a[20] = { 0 };

	fgets(a, 4, pff);
	printf("%s", a);	//打印结果 abc
	fgets(a, 5, pff);
	printf("%s", a);	//打印结果 abcdefg
	//可见,fgets 读取时,前面正常读取,最后一位会添加上'\0',再返回。
	//可见fgets读取时,会从上次'\0'的位置开始读取,但不影响源文件。
	//即,fgets读取的字符串比要求的少一位,下次读取自动从上次结束的位置开始。

	return 0;
}

fptintf函数:

//格式化输出函数 fprintf 使用
struct S	//声明结构体类型,包含一个整型i成员,一个20字符数组 c[20]成员,一个浮点数成员f
{
	int i;
	char c[20];
	float f;
};
int main()
{
	struct S s = { 1,"test",5.5 };	//定义结构体 struct S 类型变量s。
	//打开文件
	FILE* pf = fopen("test.txt", "w");	//打开文件,向其写入数据
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fprintf(pf, "%d %s %f\n", s.i, s.c, s.f);	//向pf指向的文件写入数据,这里是test.txt
	//写入数据为结构体变量的参数 i,c,f 。
	//打开文件 test.txt ,可以看到第一行内容为:1 test 5.500000
	//说明fprintf可以按要求的格式向文件写入数据。

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

fscanf函数:

//格式化输入函数 fscanf 函数使用
struct S
{
	int i;
	char c[20];
	float f;
};

int main()
{
	struct S s = { 0 };	//定义结构体变量s,内容为0.
	FILE* pf = fopen("test.txt", "r");	//打开文件,读取其中内容
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}

	fscanf(pf, "%d %s %f", &s.i, s.c, &s.f);	//按写入时的格式读取数据
	printf("%d %s %f\n", s.i, s.c, s.f);	//打印结果1 test 5.500000
	//说明fscanf 函数可以按fprintf 函数写入时的格式将数据读取出来

	fclose(pf);
	pf = NULL;

	return 0;
}

fwrite函数:

//二进制输出函数 fwrite 函数使用
struct S
{
	int i;
	char c[20];
	float f;
};
int main()
{
	struct S s = { 1,"abcde",3.3f };
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}

	fwrite(&s, sizeof(struct S), 1, pf);	//将变量s的数据以二进制形式输出到pf指向的文件。
	//打开test.txt 文件,可以看到其中数据: "   abcde               33S@"
	//因为是二进制数据,文本文件看到的是乱码

	fclose(pf);
	pf = NULL;

	return 0;
}

fread函数:

//二进制输入函数 fread 函数使用
struct S
{
	int i;
	char c[20];
	float f;
};
int main()
{
	struct S s = { 0 };
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}

	fread(&s, sizeof(struct S), 1, pf);	//以写入时的格式将pf指向文件中的数据读取出来。
	//打印结果:1 abcde 3.300000 。
	//可见fread 可以将 fwrite写入的数据读出并从二进制还原成原数据。

	printf("%d %s %f\n", s.i, s.c, s.f);

	fclose(pf);
	pf = NULL;

	return 0;
}

5、文件的非顺序读写

        5.1、fseek

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

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

fseek函数,写演示如下:

文件的非顺序读写
//fseek 函数使用

int main()
{
	FILE* pf = fopen("test.txt", "w");	//先创建一个文件,写入 abcdef 字符串
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("abcdef", pf);	//写入字符串 abcdef
	//打开文件可以看到文件中有abcdef 字符串
	fclose(pf);
	pf = NULL;

	FILE* p = fopen("test.txt", "w");
	if (p == NULL)
	{
		perror("fopen");
		return 1;
	}

	fseek(p, 3, SEEK_CUR);	//从文件当前位置向后偏移3位为起点。
	//fseek,从文件指针p指向的文件的 指定指针位置 偏移 num个字节进行操作。
	//fseek 函数第三个参数有3中设定
	//SEEK_CUR:从当前指针位置,需要清楚现在指针指向的位置再操作
	//SEEK_END:从文件末尾,所以第二个参数偏移量为负数
	//SEEK_SET:	从文件开始

	fputs("aaa", p);	//写入字符串aaa。
	//打开文件,可以看到文件中有 “   aaa”字符串。
	//可见“w”形式打开文件,是将文件直接覆盖。
	//当文件中有数据时,即使指定偏移量输出数据,也会将前面的置为“ ”。

	fclose(p);
	p = NULL;

	return 0;
}

fseek函数,读取演示如下:

  

//fseek 函数使用2
int main()
{
	FILE* pf = fopen("test.txt", "w");	//先创建一个文件,写入 abcdef 字符串
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("abcdef", pf);	//写入字符串 abcdef
	//打开文件可以看到文件中有abcdef 字符串
	fclose(pf);
	pf = NULL;

	FILE* p = fopen("test.txt", "r");	//以读取的方式打开文件
	if (p == NULL)
	{
		perror("fopen");
		return 1;
	}

	fseek(p, -3, SEEK_END);	//从文件末端位置向前偏移3位操作。

	char ch[20] = { 0 };
	fgets(ch,3, p);		//读取3个字符,并将第三个字符转化为\0,确保字符串结束。

	printf("%s\n", ch);	//打印结果 de。
	//之前例子中,如果直接读取,打印结果应该是ab。这里是de。
	//可见fseek确实将读取的指针位置从末尾向前移动了3位

	fclose(p);
	p = NULL;

	return 0;
}

      5.2、ftell

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

        long int ftell ( FILE* stream );

演示如下:

//ftell 函数使用
int main()
{
	FILE* pf = fopen("test.txt", "w");	//先创建一个文件,写入 abcdef 字符串
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("abcdef", pf);	//写入字符串 abcdef
	//打开文件可以看到文件中有abcdef 字符串
	fclose(pf);
	pf = NULL;

	FILE* p = fopen("test.txt", "r");	//以读取的方式打开文件
	if (p == NULL)
	{
		perror("fopen");
		return 1;
	}

	fseek(p, 2, SEEK_SET);	//从文件起始位置向后偏移2位操作。

	int i = ftell(p);
	printf("%d\n", i);	//打印结果 2。 
	//可见ftell 函数返回了文件指针的当前位置。可以配合fseek 的SEEK_CUR 参数使用。

	fclose(p);
	p = NULL;

	return 0;
}

        5.3、rewind

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

        void rewind ( FILE* stream );

演示如下:

//rewind 函数使用
int main()
{
	FILE* pf = fopen("test.txt", "w");	//先创建一个文件,写入 abcdef 字符串
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("abcdef", pf);	//写入字符串 abcdef
	//打开文件可以看到文件中有abcdef 字符串
	fclose(pf);
	pf = NULL;

	FILE* p = fopen("test.txt", "r");	//以读取的方式打开文件
	if (p == NULL)
	{
		perror("fopen");
		return 1;
	}

	char ch[20] = { 0 };
	fgets(ch, 3, p);		//读取前3个字符,并将读取来的第三个字符置为结束符 '\0'。
	printf("%s\n", ch);		//打印结果 ab 。
	//此时文件指针指向第3位,继续打印会从第3位开始。

	char ch1[20] = { 0 };
	fgets(ch1, 3, p);		//读取3个字符,并将读取来的第三个字符置为结束符 '\0'。
	printf("%s\n", ch1);		//打印结果 cd 。
	//可见文件指针确实在向后顺序移动。

	rewind(p);				//将文件指针置为初始位置
	char ch2[20] = { 0 };
	fgets(ch2, 3, p);		
	printf("%s\n", ch2);	//打印结果 ab 。
	//可见rewind 确实将文件指针置为初始位置。

	fclose(p);
	p = NULL;

	return 0;
}

6、文本文件和二进制文件

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

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

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

        数据在内存中的存储:

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

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

演示如下:

//文本文件和二进制文件
//以10000为例
//如果是按文本文件存储,那它是五个字符 '1''0''0''0''0',占5个字节。
//如果是按二进制存储,那它是一个整型,4个字节:0000 0000 0000 0000 0010 0111 0001 0000
int main()
{
	FILE* pf = fopen("test.txt", "wb");	//以二进制写入的方式打开文件
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	int i = 10000;
	fwrite( &i, sizeof(int), 1, pf);	//以二进制形式将整型 10000 写入pf指向的文件。
	//以文本形式打开文件(txt格式),可以看到一个乱码 “'  ”。
	//文本文件中的字符都是ASCII码的形式
	//以文本形式打开二进制文件,它会以ASCII码形式解读,就出现乱码了。

	//VS2019提供了二进制形式打开文件
	//先找到解决方案资源管理器,就是放着头文件、源文件的小窗口
	//如果找不到,可以点击视图->解决方案资源管理器(Ctrl+Alt+L)
	//鼠标右键点击源文件->添加->现有项->选择文件->添加
	//这里我打开的是解决方案下面的test.txt文件
	//这时可以看到test.txt 文件添加到源文件列表下方。
	//右击文件->打开方式->二进制编辑器->确定

	//可以看到文件中保存着 10 27 00 00 数据。
	// 
	//二进制文件是以十六进制显示的,将其还原得到
	//  1    0   2     7    0   0    0     0
	//0001 0000 0010 0111 0000 0000 0000 0000
	//10000 整型按4个字节存储是
	//0000 0000 0000 0000 0010 0111 0001 0000,解读成十六进制
	//  0   0     0    0    2    7    1    0
	//00 00 27 10
	//可以看出,二进制文件就是数据在小端存储内存上的形式。

	fclose(pf);
	pf = NULL;

	return 0;
}

补充:sprintf 和 sscanf 函数使用演示如下:

//sprintf 和 sscanf 函数使用
int main()
{
	int a = 12;
	char tmp[20] = { 0 };
	sprintf(tmp, "%d", a);

	printf("%s", tmp);	//以字符串形式打印tmp ,打印结果 12。
	//说明整型12被转化成了字符串。
	int b = 0;
	sscanf(tmp, "%d", &b);
	printf("%d", b);	//以整型形式打印变量b,打印结果 12。
	//说明tmp字符串中的12被转换成整型了。

	return 0;
}

7、文件读取结束的判定

        7.1、被错误使用的feof

        int feof ( FILE* stream );

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

        feof函数是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。

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

                1)fgetc 读取错误或读取结束时会返回EOF。

                2)fgets 读取错误或读取结束会返回NULL。

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

                1)fread 读取错误或读取接货会返回小于实际要读的个数。

演示如下:

//feof 函数使用
//当文件读取结束时,可以通过feof返回值判断是文件结束还是读取失败。
int main()
{
	FILE* pf = fopen("test.txt", "w");	//先创建一个文件,写入 abcdef 字符串
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("abcdef", pf);	//写入字符串 abcdef
	//打开文件可以看到文件中有abcdef 字符串
	fclose(pf);
	pf = NULL;

	FILE* p = fopen("test.txt", "r");
	if (p == NULL)
	{
		perror("fopen");
		return 1;
	}

	char ch = '0';
	//这里使用fgetc来测试,fgetc读取错误或结束时会返回EOF
	while ((ch = fgetc(p)) != EOF)	//当读取不为 EOF 时,继续读取。
	{
		printf("%c",ch);	//打印结果 abcdef
	}
	printf("\n");

	//fgetc 停止读取,可能是读取结束或遇到错误,用feof进行判断

	//feof会尝试向后继续读取信息,如果这里是文件结束,则返回非零值。
	if( feof(p) != 0 )
	{
		printf("文件正常结束!\n");	//打印结果 文件正常结束。
	}
	else
	{
		printf("文件读取失败结束!\n");
	}

	fclose(p);
	p = NULL;

	return 0;
}

8、文件缓冲区

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

        因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区的操作,或者在文件操作结束的时候关闭文件。如果不做,可能导致读写文件的问题。

  • 14
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值