C语言中,这样规定了一种文件指针,文件指针的作用就是能够很方便的找到我们要读写的文件。
在VS2013中的<stdio.h>头文件中,文件类型的声明如下
struct _iobuf {
char *_ptr;
int _cnt;
char *_base;
int _flag;
int _file;
int _charbuf;
int _bufsiz;
char *_tmpfname;
};
typedef struct _iobuf FILE;
可以很明确看出,这是一个结构体,并且结构体重命名为 FILE,那么我们在使用的时候就可以这样定义一个文件指针:
FILE* pf;//文件指针变量
和其他指针相似,每当我们打开一个新的文件的时候,都需要一个新的指针变量来指向对应的文件。
文件的打开和关闭
有了文件指针的概念,那么之后就是要明白在文件读写之前,需要先打开文件,在读写结束的时候,需要关闭对应的文件以免造成内存空间的占用。那么文件指针就相当于是一个媒介,方便我们很准确的找到对应的文件并打开。
C语言中文件的打开和关闭可以说是一对出现的函数分别是:fopen 和 fclose
//打开文件
FILE * fopen ( const char * filename, const char * mode );
//关闭文件
int fclose ( FILE * stream );
在这里,filename 是字符串,用来命名文件,访问模式 mode 的值可以是下列值中的一个:
这里附带上菜鸟教程的连接https://www.runoob.com/cprogramming/c-file-io.html
简单举几个常见的 mode 的例子代码:
#include <stdio.h>
int main()
{
FILE* pf;//创建一个文件指针
pf = fopen("test.txt", "w");//打开一个文本文件,进行写入操作
fclose(pf);
pf = fopen("test.txt", "r");//进行只读操作
fclose(pf);
return 0;
}
文件的输入和输出
文件打开的方式,我们已经知道了,那么打开文件之后,如何对文件进行输入和输出的操作呢。
这里首先要明白一点,我们说的输入时,表示的是我们从键盘上敲打一些内容,并将内容存储到文件中。而输出的意思则是,文件的内容要在屏幕上显示出来。有了这样概念,我们再来看看C语言中的几种函数所提供的输入和输出:
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输出流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出 | fwrite | 文件 |
还是简单的举一个例子:
#include <stdio.h>
int main()
{
FILE* pf;//创建一个文件指针
char ch[] = "hello,bit!!";
pf = fopen("test.txt", "wb");//打开一个文本文件,进行写入操作
fwrite(ch, sizeof(ch), 1, pf);
fclose(pf);
return 0;
}
代码运行之后的结构是我们向 test.txt 的文件中以二进制的形式写入了 hello,bit!!
文本文件和二进制文件
根据数据的组织形式,数据文件被称为文本文件或者二进制文件。
数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。 如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文 本文件。 一个数据在内存中是怎么存储的呢? 字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。
文件读取结束的判定
牢记:在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。 而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
1. 文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets ) 例如: fgetc 判断是否为 EOF . fgets 判断返回值是否为 NULL .
2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。 例如: fread判断返回值是否小于实际要读的个数。
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int c; // 注意:int,非char,要求处理EOF
FILE* fp = fopen("test.txt", "r");
if(!fp) {
perror("File opening failed");
return EXIT_FAILURE;
}
//fgetc 当读取失败的时候或者遇到文件结束的时候,都会返回EOF
while ((c = fgetc(fp)) != EOF) // 标准C I/O读取文件循环
{
putchar(c);
}
//判断是什么原因结束的
if (ferror(fp))
puts("I/O error when reading");
else if (feof(fp))
puts("End of file reached successfully");
fclose(fp);
}
#include <stdio.h>
enum { SIZE = 5 };
int main(void)
{
double a[SIZE] = {1.,2.,3.,4.,5.};
FILE *fp = fopen("test.bin", "wb"); // 必须用二进制模式
fwrite(a, sizeof *a, SIZE, fp); // 写 double 的数组
fclose(fp);
double b[SIZE];
fp = fopen("test.bin","rb");
size_t ret_code = fread(b, sizeof *b, SIZE, fp); // 读 double 的数组
if(ret_code == SIZE) {
puts("Array read successfully, contents: ");
for(int n = 0; n < SIZE; ++n) printf("%f ", b[n]);
putchar('\n');
} else { // error handling
if (feof(fp))
printf("Error reading test.bin: unexpected end of file\n");
else if (ferror(fp)) {
perror("Error reading test.bin");
}
}
fclose(fp);
}
文件缓冲区
还有一个很重要的概念,那就是文件缓冲区,对于文件缓冲区来说,只需要注意一点,那就是C语言只有在刷新缓冲区或者文件关闭的时候,才会把缓冲区的内容写入文件中,如果不做那就对出现问题。