文件
程序文件
源程序文件(后缀为.c)
目标文件(windows环境后缀为.obj)
可执行程序(windows环境后缀为.exe)
数据文件
程序运行时从中读取数据或输出内容的文件
文件名
一个文件要有唯一的文件标识,文件标识常被称为文件名。
文件名包含三部分:
文件路径+文件名主干+文件后缀
例:c:\code\test.txt
文件指针
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名 字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是有系统 声明的,取名FILE
FILE* pf; <----pf为文件指针变量
通过文件指针变量能够找到与它关联的文件。
文件的打开和关闭
FILE * fopen ( const char * filename, const char * mode );
int fclose ( FILE * stream );
文件使用方式 | 含义 | 如果指定文件不存在 |
---|
“r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 | “w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 | “a”(追加) | 向文本文件尾添加数据 | 建立一个新的文件 | “rb”(只读) | 为了输入数据,打开一个二进制文件 | 出错 | “wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新的文件 | “ab”(追加) | 向一个二进制文件尾添加数据 | 出错 | “r+”(读写) | 为了读和写,打开一个文本文件 | 出错 | “w+”(读写) | 为了读和写,建立一个新的文件 | 建立一个新的文件 | “a+”(读写) | 打开一个文件,在文件尾进行读写 | 建立一个新的文件 | “rb+”(读写) | 为了读和写打开一个二进制文件 | 出错 | “wb+”(读写) | 为了读和写,新建一个新的二进制文件 | 建立一个新的文件 | “ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 建立一个新的文件 |
例子🌰
#include <stdio.h>
int main()
{
FILE* pf = fopen("test.txt", "w");
if (pf != NULL)
{
fputs("abc", pf);
fclose(pf);
pf = NULL;
}
return 0;
}
文件的顺序读写
功能 | 函数名 | 适用于 |
---|
字符输入函数 | fgetc | 所有输入流 | 字符输出函数 | fputc | 所有输出流 | 文本行输入函数 | fgets | 所有输入流 | 文本行输出函数 | fputs | 所有输出流 | 格式化输入函数 | fscanf | 所有输入流 | 格式化输出函数 | fprintf | 所有输出流 | 二进制输入 | fread | 文件 | 二进制输出 | fwrite | 文件 |
字符输入输出(fgetc, fputc)
把文件A的内容复制到文件B
#include <stdio.h>
#include <string.h>
int main()
{
FILE* pf1 = fopen("A.txt", "r");
FILE* pf2 = fopen("B.txt", "w");
if (pf1 == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
char ch = 0;
while ((ch = fgetc(pf1)) != EOF)
{
fputc(ch, pf2);
}
fclose(pf1);
pf1 = NULL;
fclose(pf2);
pf2 = NULL;
return 0;
}
格式化输入输出(fscanf, fprintf)
将结构体数据写到文件
#include <stdio.h>
#include <string.h>
struct Stu
{
char name[20];
int age;
double sc;
};
int main()
{
struct Stu s = { "老王",19,86.5 };
FILE* pf = fopen("data.txt", "w");
if (pf == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
fprintf(pf, "%s %d %lf", s.name, s.age, s.sc);
fclose(pf);
pf = NULL;
return 0;
}
从文件中读取数据
将文件中的数据读取到结构体s2中,打印s2的数据
int main()
{
struct Stu s = { "老王",19,86.5 };
struct Stu s2 = { 0 };
FILE* pf = fopen("data.txt", "r");
if (pf == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
fscanf(pf, "%s %d %lf", s2.name, &s2.age, &s2.sc);
printf("%s %d %lf\n", s2.name, s2.age, s2.sc);
fclose(pf);
pf = NULL;
return 0;
}
- 注意:fscanf和fprintf 其实只是比scanf和printf多一个参数—流
二进制输入输出(fread, fwrite)
size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );
size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );
例子🌰
int main()
{
struct Stu s[2] = { { "老王",19,86.5 },{"老张",18,89.9} };
struct Stu s2[2] = { 0 };
FILE* pf = fopen("data.txt", "w");
if (pf == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
fwrite(&s, sizeof(struct Stu), 2, pf);
fclose(pf);
pf = NULL;
return 0;
}
按照二进制写出来的文件采用文本形式打开是乱码。因为二进制不是给人看的,所以二进制写的应该用二进制读。
int main()
{
struct Stu s[2] = { { "老王",19,86.5 },{"老张",18,89.9} };
struct Stu s2[2] = { 0 };
FILE* pf = fopen("data.txt", "r");
if (pf == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
fread(&s2, sizeof(struct Stu), 2, pf);
for (int i = 0; i < 2; i++)
{
printf("%s %d %lf\n", s2[i].name, s2[i].age, s2[i].sc);
}
fclose(pf);
pf = NULL;
return 0;
}
应用:[C语言] 通讯录|静态 动态 文件 链表 多版本讲解_CegghnnoR的博客-CSDN博客
流的概念
流是磁盘或其它外围设备中存储的数据的源点或终点。
C语言中对流除了分为I/O流之外,还分为文本流与二进制流。
int main()
{
fputs("abc", stdout);
return 0;
}
除了fread和fwrite,以上表格中的函数都适用于所有的输入流或输出流,所以它们也可以使用stdin或stdout来从键盘读取数据或向屏幕输出数据,而scanf和printf只是针对标准输入输出流的函数。
补充:sscanf, sprintf
sprintf: 将格式化数据转换为字符串,写进字符数组里。
sscanf: 将字符串写成格式化数据。
例子🌰
int main()
{
struct Stu s = { "老王",19,86.5 };
char buf[100] = { 0 };
sprintf(buf, "%s %d %lf", s.name, s.age, s.sc);
printf("%s\n", buf);
struct Stu s2 = { 0 };
sscanf(buf, "%s %d %lf", s2.name, &s2.age, &s2.sc);
printf("%s %d %lf\n", s2.name, s2.age, s2.sc);
return 0;
}
总结:
-
scanf和printf只是针对标准输入输出流。 -
fscanf和fprintf适用所有输入输出流。 -
sscanf和sprintf针对字符串。
文件的随机读写
若data文件内容为abc
int main()
{
FILE* pf = fopen("data.txt", "r");
if (pf == NULL)
{
printf("%s", strerror(errno));
return 0;
}
char ch = 0;
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
fclose(pf);
pf = NULL;
return 0;
}
输出结果为:
a
b
c
说明每调用一次fgetc,文件指针就向后移一位。
而有时候我们需要自己去移动文件指针
int fseek ( FILE * stream, long int offset, int origin );
- 根据文件指针的位置和偏移量来定位文件指针
- offset:偏移量(+值向后,-值向前)
- origin:基准位置(有三个选择)
Constant | 基准位置 |
---|
SEEK_SET | 文件开始 | SEEK_CUR | 文件指针当前位置 | SEEK_END | 文件末尾 |
data文件内容为abcdef
int main()
{
FILE* pf = fopen("data.txt", "r");
if (pf == NULL)
{
printf("%s", strerror(errno));
return 0;
}
char ch = 0;
ch = fgetc(pf);
printf("%c", ch);
ch = fgetc(pf);
printf("%c", ch);
fseek(pf, 3, SEEK_CUR);
ch = fgetc(pf);
printf("%c", ch);
fclose(pf);
pf = NULL;
return 0;
}
结果为abf
fseek将文件指针向后移动了3位,跳过了cde。
long int ftell ( FILE * stream );
返回文件指针相对于起始位置的偏移量
void rewind ( FILE * stream );
让文件指针回到起始位置
rewind(pf); 相当于fseek(pf, 0, SEEK_SET);
文本读取结束时的判定:feof
当文件读取结束的时候,判断是读取失败结束还是遇到文件尾结束
要判断文件是否读取结束,要看以下几点:
- 文本文件读取结束
fgetc 返回值为EOF fgets 返回值为NULL - 二进制文件读取结束
例子🌰
#include <stdio.h>
#include <stdlib.h>
int main()
{
int c;
FILE* fp = fopen("test.txt", "r");
if (!fp)
{
perror("File opening failed");
return EXIT_FAILURE;
}
while ((c = fgetc(fp)) != EOF)
{
putchar(c);
}
if (ferror(fp))
puts("I/O error when reading");
else if (feof(fp))
puts("End of file reached successfully");
fclose(fp);
return 0;
}
- ferror返回真表示出现I/O错误
- feof返回真表示遇到文件尾结束
|