前言
在文件操作(上卷)中,讲到的主要都是正式文件操作开始之前的前置知识,而这一卷中,我们将开始正式地操作文件。
在上卷中我们已经说到,stdin stdout stderr是三个C语言程序启动时默认打开的流。这三个流的类型是:FILE*,通常称为文件指针。而C语言,就是通过FILE*的文件指针来维护流的各种操作的。
文件指针
缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”。
每个被使用的文件(包括外部设备)都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。当我们打开一个文件时,就会创造一个和这个文件相关的文件信息区。
这些信息保存在一个结构体变量中。该结构体类型是由系统声明的,取名FILE。
比如图中,我们现在有一个data.txt文件,操作这个文件就会创造一个FILE类型的文件信息区f,文件信息区和我们的文件建立了一定联系,里面放着文件相关的各种信息。
在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;//使用typedef重命名
可以看到我们的FILE其实是对一个结构体类型的重命名,FILE就是这样一个结构体类型。
所以我们创建的文件信息区就是创建的一个结构体变量。
注意,不同的C编译器的FILE类型包含的内容不完全相同,但大同小异。
每当打开一个文件时,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者则不必关心细节。
一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。
我们的文件信息区是有地址的,我们将其地址放到一个类型为FILE*的指针中去。
我们上面说stdin stdout stderr这三个流的类型是:FILE*,现在我们知道,其实它们就是指向FILE类型文件信息区的三个指针。
每个文件都有自己的文件信息区,文件是在硬盘上放着的,文件信息区是在内存上放着的。
文件的打开和关闭
文件的操作:
1.打开文件---打开流
2.读写文件---读/写流
3.关闭文件---关闭流
流的底层是怎么做的,不用关心。
ANSI C规定:使用fopen函数来打开文件,fclose来关闭文件。
fopen
第一个参数是文件名,后一个参数是文件的打开方式。
最终返回的是文件信息的地址。用它可以找到文件信息区,就可以找到流。
mode表示文件的打开模式,有很多种,以下都是:
注意是双引号。
文件操作示意图:
"w"(只读),如果指定的文件存在,会把文件内容清洗掉,然后展示出一个空的文件;如果指定的文件不存在,就会建立一个新的文件。
演示:
可以看到,我们就生成了一个data.txt,大小为0。
现在我们在data.txt里写一点东西:
那么当我们再次运行程序,此时是已存在这个文件,会发现运行完再打开这个文件发现已经被改为空白了:
这就是"w"(只读)的特点,如果指定的文件存在,会把文件内容清洗掉,然后展示出一个空的文件;如果指定的文件不存在,就会建立一个新的文件。
如果是"r"的方式打开,现在我们还是打开data.txt,因为存在,所以运行起来不会报错。
如果我们把这个文件删除然后再运行:
这时就给出了相应的报错信息:没有这样的文件或目录。
打开文件后我们读文件或写文件,然后我们就要关闭文件。
fclose
所以我们就这样关闭文件:
但是流关掉后,我们的pf的地址其实还在,不会自己置空,避免变为野指针,我们要置为空:
打开其他位置的文件
可以发现,我们现在打开的文件必须在当前工程路径底下,那如果现在我们想打开桌面放置的一个文件呢?
那就加上路径。路径在哪看?右击这个文件点开属性:
可以看到位置一行,后面的就是路径,所以我们这样写:
记得将\写为\\,否则会被当做转义字符。
这种从“根”上开始的写法叫做绝对路径。
此外,还有一种相对路径的写法:
比如,我们可以用来打开上一级路径下的文件:
这是我们的工程目录:
(可以理解为test.c所在的这一个目录)
所以这就是上一级路径:
.表示当前路径,..表示上一级路径,所以我们的代码这样写:
FILE* pf = fopen(".\\..\\data.txt", "w");
这就是,当前路径的上一级路径底下的data.txt文件。
如果我们想要打开上一级的上一级路径下的文件,就这样写:
FILE* pf = fopen(".\\..\\..\\data.txt", "w");
这就是,当前路径的上一级的上一级路径底下的data.txt文件。
从这种写法可以看出,相对路径所谓的相对,就是相对于当前位置的路径的意思。
文件的顺序读写
有两种读写,一种叫顺序读写,一种叫随机读写。
顺序读写函数介绍
fgetc也就是读字符。fputc是写字符。fgets是读字符串,fputs是写字符串。
文本行就是多个字符的意思。
fscanf和fprintf是格式化输入和输出。
fread和fwrite是二进制输入和输出。
前面的6个函数读和写的都是文本信息(也就是肉眼能看得懂),而后面两个读和写的是二进制的信息。
fputc函数
前一个参数是要写的字符,后一个参数是流 。
现在我们可以把字母a~z写到test.txt中去。
fgetc函数
只有一个参数,就是流(文件指针),返回一个int。
成功情况返回的是读到字符的ASCII码值。
如果读取失败,会返回一个EOF。也就是end of file,文件的结束标志,其实设置在全局变量中。 可以写下EOF右击转到定义,会看到其实本质就是-1。
那么现在我们可以试着读一下我们刚才写的字符:
其实在我们读或者写的时候,文件都有一个光标在按顺序移动,否则读和写就乱套了。
我们还可以这样写,把所有字符都读取:
补充:fgetc fputc的适用范围
这时我们还注意到,上面说到,前6个函数适用于所有输入输出流 ,这是个什么情况呢?
当我们以读的形式打开文件,得到的是文件的输入流;当我们以写的形式打开文件,得到的是文件的输出流。
stdin stdout也是输入和输出流,那么我们前面提到的两个函数也能适用于这两个标准流。
所以这两个函数可以适用于标准输出输入流或文件输出输入流。
到此,本文内容就结束了,后面会继续补充文件操作相关的内容,祝阅读愉快^_^