引言:谈起C语言,大家首先想到的是程序,但是你是否想过我们写的代码是如何被计算机执行起来的呢?下面我将就这一问题展开接下来的内容。
一.文件
1.为何要使用文件?
若没有文件,我们所写的程序是以数据的形式存储在内存中,当退出程序时,内存被回收,数据就丢失了。所以,如果想将数据进行永久保存,我们就可以使用文件。
2.文件的位置:文件在磁盘(硬盘)上
3.文件名:文件路径+文件名主干+文件后缀
注:文件后缀不是必须要有的
4.文件的分类(按文件功能分)
1>程序文件:包括源文件(.c),目标文件(在Windows环境下为.obj),可执行文件(在Windows环境下 为.exe)
2>数据文件:根据数据的组织形式,将数据文件被称为二进制文件和文本文件
2.1>数据在内存中以二进制的形式存储,若不加转换的输出到外存的文件中,就是二进制文件; 若加以转换,以ASCII码的形式存储在外存上,则为文本文件。
2.2>字符一律以ASCII码的形式存储在文件中,
数值型的数据既可以以使用ASCII码形式存储,也可以二进制的形式存储
(注:究竟以怎样的方式存储数值型数据更节省内存,是要看具体的数据的。
例如100,若以二进制形式存储需要4个字节,若以ASCII码的形式存储需要3个字节)
二.有关文件的操作
1.文件的打开与关闭(打开与关闭需一一对应)
1>流:为了方便描述数据的来去,我们抽象出流的概念
2>文件指针:每当我们打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构体的变量,并填充其中的信息,这个时候一般是通过一个FILE的指针来维护这个FILE结构体变量
3>打开文件:FILE* fopen(const char* filename, const char* mode);
fopen - C++ Reference (cplusplus.com)
mode:文件打开方式
文件使用方式 | 含义 | 如果指定的文件不存在 |
"r"(只读) | 为输入数据打开一个已经存在的文本文件 | 出错 |
"w"(只写) | 为输出数据,打开一个文本文件 | 建立一个新文件 |
"a"(追加) | 向文本文件尾追加数据 | 建立一个新文件 |
"rb"(只读) | 为了输入数据,打开一个二进制文本文件 | 出错 |
"wb"(只写) | 为了输出数据,打开一个二进制文本文件 | 建立一个新文件 |
"ab"(追加) | 向二进制文件尾为追加数据 | 建立一个新文件 |
"r+"(读写) | 为了读和写,打开一个文本文件 | 出错 |
"w+"(读写) | 为了读和写,建立一个文本文件 | 建立一个新文件 |
"a+"(读写) | 打开一个文本文件,在文件尾进行读写 | 建立一个新文件 |
"rb+"(读写) | 为了读和写,打开一个二进制文件 | 出错 |
"wb+"(读写) | 为了读和写,建立一个二进制文件 | 建立一个新文件 |
"ab+"(读写) | 打开一个二进制文件,在文件尾进行读写 | 建立一个新文件 |
4>关闭文件:int fclose(FILE* stream);
fclose - C++ Reference (cplusplus.com)
2.文件的顺序读写
1>
函数名 | 功能 | 适用于 |
fgetc | 字符输入函数 | 所有输入流 |
fputc | 字符输出函数 | 所有输出流 |
fgets | 文本行输入函数 | 所有输入流 |
fputs | 文本行输出函数 | 所有输出流 |
fscanf | 格式化输入函数 | 所有输入流 |
fprintf | 格式化输出函数 | 所有输出流 |
fread | 二进制输入 | 文件输入流 |
fwrite | 二进制输出 | 文件输出流 |
fgetc:int fgetc ( FILE * stream );
#include <stdio.h>
int main()
{
FILE* pFile;
int c;
int n = 0;
pFile = fopen("myfile.txt", "r");
if (pFile == NULL) perror("Error opening file");
else
{
do {
c = fgetc(pFile);
if (c == '$') n++;
} while (c != EOF);
fclose(pFile);
printf("The file contains %d dollar sign characters ($).\n", n);
}
return 0;
}
其他的函数就不在此一一展开说了,若有兴趣可以在cplusplus.com上自己查找
2>对比一组函数
3.文件的随机读写
1>fseek
作用:根据文件指针的位置和偏移量来定义文件指针
函数原型:int fseek(FILE* stream, long int offset, int origin);
对于该函数的起始位置有三种情况:
SEEK_SET:文件起始位置
SEEK_CUR:文件指针当前位置
SEEK_END:文件末尾
2>ftell
含义:返回文件指针相对于起始位置的偏移量
函数原型:long int ftell(FILE* stream);
3>rewind
含义:让文件指针回到文件的起始位置
函数原型:void rewind(FILE* stream);
4.文件读取错误的判定
1>feof:遇到文件末尾而结束
2>ferror:读取时发生错误而结束
注:对于不同的函数当其结束时返回的值各不相同,需要我们自己去记忆
3>对于二进制文件读取是否错误的判定是根据返回值是否小于实际要读的数的个数
三.文件的缓冲区
为了提高效率,从内存向磁盘传输的数据是先传到缓冲区,待缓冲区装满后再一起送到磁盘上