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.文件的随机读写
下面,我们来继续介绍几个有关文件随机读写的函数,fseek、ftell以及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;
}
运行结果见图九: