上一章我们讲解了 C 语言处理的标准输入和输出设备。本章我们将介绍 C 程序员如何创建、打开、关闭文本文件或二进制文件。 一个文件,无论它是文本文件还是二进制文件,都是代表了一系列的字节。C 语言不仅提供了访问顶层的函数,也提供了底层(OS)调用来处理存储设备上的文件。本章将讲解文件管理的重要调用。

打开文件

您可以使用fopen( )函数来创建一个新的文件或者打开一个已有的文件,这个调用会初始化类型FILE的一个对象,类型FILE包含了所有用来控制流的必要的信息。下面是这个函数调用的原型:

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

在这里,filename是字符串,用来命名文件,访问模式mode的值可以是下列值中的一个:

模式 描述
r 打开一个已有的文本文件,允许读取文件。
w 打开一个文本文件,允许写入文件。如果文件不存在,则会创建一个新文件。在这里,您的程序会从文件的开头写入内容。如果文件存在,则该会被截断为零长度,重新写入。
a 打开一个文本文件,以追加模式写入文件。如果文件不存在,则会创建一个新文件。在这里,您的程序会在已有的文件内容中追加内容。
r+ 打开一个文本文件,允许读写文件。
w+ 打开一个文本文件,允许读写文件。如果文件已存在,则文件会被截断为零长度,如果文件不存在,则会创建一个新文件。
a+ 打开一个文本文件,允许读写文件。如果文件不存在,则会创建一个新文件。读取会从文件的开头开始,写入则只能是追加模式。

如果处理的是二进制文件,则需使用下面的访问模式来取代上面的访问模式:

  1. "rb", "wb", "ab", "rb+", "r+b", "wb+", "w+b", "ab+", "a+b"

关闭文件

为了关闭文件,请使用 fclose( ) 函数。函数的原型如下:

  1. int fclose( FILE *fp );

如果成功关闭文件,fclose( )函数返回零,如果关闭文件时发生错误,函数返回EOF。这个函数实际上,会清空缓冲区中的数据,关闭文件,并释放用于该文件的所有内存。EOF 是一个定义在头文件stdio.h中的常量。
C 标准库提供了各种函数来按字符或者以固定长度字符串的形式读写文件。

写入文件

下面是把字符写入到流中的最简单的函数:

  1. int fputc( int c, FILE *fp );

函数fputc()把参数 c 的字符值写入到 fp 所指向的输出流中。如果写入成功,它会返回写入的字符,如果发生错误,则会返回EOF。您可以使用下面的函数来把一个以 null 结尾的字符串写入到流中:

  1. int fputs( const char *s, FILE *fp );

函数fputs()把字符串s写入到 fp 所指向的输出流中。
如果写入成功,它会返回一个非负值,
如果发生错误,则会返回EOF
您也可以使用 fprintf 函数把一个字符串写入到文件中

  1. int fprintf(FILE *fp,const char *format, ...)

⚠️注意:请确保您有可用的tmp目录,如果不存在该目录,则需要在您的计算机上先创建该目录。 /tmp一般是 Linux 系统上的临时目录,如果你在 Windows 系统上运行,则需要修改为本地环境中已存在的目录,例如:C:\tmp、D:\tmp等。

尝试下面的实例:

  1. #include <stdio.h>
  2. int main()
  3. {
  4. FILE *fp = NULL;
  5. fp = fopen("/tmp/test.txt", "w+");
  6. fprintf(fp, "This is testing for fprintf...\n");
  7. fputs("我", fp);
  8. fputs("是", fp);
  9. fputs("一缕清风\n", fp);
  10. fputs("你呢?\n", fp);
  11. fclose(fp);
  12. }

image.png
当上面的代码被编译和执行时,它会在 /tmp 目录中创建一个新的文件 test.txt,并使用两个不同的函数写入两行。接下来让我们来读取这个文件。

读取文件

下面是从文件读取单个字符的最简单的函数:

  1. int fgetc( FILE * fp );

fgetc()函数从 fp 所指向的输入文件中读取一个字符。返回值是读取的字符,如果发生错误则返回EOF。下面的函数允许您从流中读取一个字符串:

  1. char *fgets( char *buf, int n, FILE *fp );

函数fgets()从 fp 所指向的输入流中读取 n - 1 个字符。它会把读取的字符串复制到缓冲区buf,并在最后追加一个null字符来终止字符串。
如果这个函数在读取最后一个字符之前就遇到一个换行符 ‘\n’ 或文件的末尾 EOF,则只会返回读取到的字符,包括换行符。您也可以使用int fscanf(FILE fp, const char format, …)函数来从文件中读取字符串,但是在遇到第一个空格和换行符时,它会停止读取。

  1. #include <stdio.h>
  2. int main()
  3. {
  4. FILE *fp = NULL;
  5. char buff[255];
  6. fp = fopen("/tmp/test.txt", "r");
  7. fscanf(fp, "%s", buff);
  8. printf("1: %s\n", buff);
  9. fgets(buff, 255, (FILE*)fp);
  10. printf("2: %s", buff );
  11. fgets(buff, 255, (FILE*)fp);
  12. printf("3: %s", buff );
  13. fgets(buff, 255, (FILE*)fp);
  14. printf("4: %s", buff );
  15. fclose(fp);
  16. }

当上面的代码被编译和执行时,它会读取上一部分创建的文件,产生下列结果:

  1. 1: This
  2. 2: is testing for fprintf...
  3. 3: 我是一缕清风
  4. 4: 你呢?

首先,fscanf()方法只读取了This,因为它在后边遇到了一个空格。其次,调用fgets()读取剩余的部分,直到行尾。最后,调用fgets()完整地读取第二行。

二进制 I/O 函数

下面两个函数用于二进制输入和输出:

  1. size_t fread(void *ptr, size_t size_of_elements,
  2. size_t number_of_elements, FILE *a_file);
  3. size_t fwrite(const void *ptr, size_t size_of_elements,
  4. size_t number_of_elements, FILE *a_file);

这两个函数都是用于存储块的读写 - 通常是数组或结构体。