C语言文件操作详解
什么是文件
文件名
文件类型
文件缓冲区
文件的打开和关闭
文件的顺序读写
文件结束的判定
一、什么是文件
磁盘上的文件是文件。
但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件
程序文件
包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境后缀
为.exe)。
数据文件
文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。
二、文件名
一个文件要有一个唯一的文件标识,以便用户识别和引用。
文件名包含3部分:文件路径+文件名主干+文件后缀。例如: c:\code\test.txt
为了方便起见,文件标识常被称为文件名。
三、文件类型
根据数据的组织形式,数据文件被称为文本文件或者二进制文件。
数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。
如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件。
一个数据在内存中是怎么存储的呢?
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。
四、文件缓冲区
ANSIC 标准采用“缓冲文件系统”处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在
使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。
五、文件的打开和关闭
文件在读写之前应该先打开文件,在使用结束之后应该关闭文件。在编写程序的时候,在打开文件的同时,都会返回一个FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系
使用fopen函数来打开文件,fclose来关闭文件。
文件的打开方式如下:
例
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <errno.h>
#include <string.h>
//
int main()
{
// //fopen函数如果是以写的形式打开
// //如果文件不存在,会创建这个文件
// //如果文件存在,会清空文件的内容
// //fopen函数如果是以读的形式打开
// //文件不存在打开失败
//
FILE* pf = fopen("data.txt", "w");
if (pf == NULL)
{
printf("%s\n", strerror(errno));
return 1;//失败返回
}
//写文件
int i = 0;
for (i = 'a'; i <= 'z'; i++)
{
fputc(i, pf);//字符输出
}
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
六、文件的顺序读写
fputc字符输出函数,每次输出一个字符
fputc/fgetc演示
fputc将字符写道data.txt文件中,fgetc从文件中,读取字符
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <errno.h>
#include <string.h>
// 文件操作.
//字符输出函数
//#include <stdio.h>
//int main()
//{
// fputc('a', stdout);
// fputc('b', stdout);
// fputc('c', stdout);
//
// return 0;
//}
int main()
{
//fopen函数如果是以写的形式打开
//如果文件不存在,会创建这个文件
//如果文件存在,会清空文件的内容
//fopen函数如果是以读的形式打开
//文件不存在打开失败
//以只写的方式打开data,txt文件
FILE* pf1 = fopen("data.txt", "w");
if (pf1 == NULL)
{
printf("%s\n", strerror(errno));
return 1;//失败返回
}
//写文件
int i = 0;
for (i = 'a'; i <= 'z'; i++)
{
//写入a-z之间的小写字母
fputc(i, pf1);
}
// //关闭文件
fclose(pf1); //关闭文件
pf1 = NULL; //文件置空
//以只读的方式打开文件
FILE* pf2 = fopen("data.txt", "r");
if (pf2 == NULL)
{
//若打开失败,返回失败原因
printf("%s\n", strerror(errno));
return 1;
}
//打开文件成功,读文件
//int ch = fgetc(pf);
//printf("%c\n", ch);//a
//ch = fgetc(pf);
//printf("%c\n", ch);//b
//ch = fgetc(pf);
//printf("%c\n", ch);//c
//ch = fgetc(pf);
//printf("%c\n", ch);//d
int ch = 0;
while ((ch = fgetc(pf2)) != EOF)
{
printf("%c ", ch);
}
//关闭文件
fclose(pf2);//关闭文件
pf2 = NULL; //指针置空
return 0;
}
对比一组函数:
scanf/fscanf/sscanf
printf/fprintf/sprintf
scanf/printf 是针对标准输入流/标准输出流格式化输入/输出语句。
fscanf/fprintf 是针对所有输入流/所有输出流的格式化输入/输出语句。
sscanf/sprintf sscanf 是从字符串中读取格式化的数据 sprintf 是把格式化数据输出成(存储到)字符串。
stdin–是标准输入流设备-从键盘输入
stdou–是标准输出设备-显示到屏幕上
fscanf/fprintf格式化输出与输出
struct Stu
{
char name[20];
int age;
float score;
};
int main()
{
struct Stu s = {"zhangsan", 20, 66.5f};
FILE* pf1 = fopen("data.txt", "w");
if (pf1 == NULL)
{
printf("%s\n", strerror(errno));
return 1;
}
//格式化的写入到文件中
fprintf(pf1,"%s %d %f", s.name, s.age, s.score);
fclose(pf1);//关闭文件
pf1 = NULL;
struct Stu s1 = { 0 };
FILE* pf2 = fopen("data.txt", "r");
if (pf2 == NULL)
{
printf("%s\n", strerror(errno));
return 1;
}
//格式化的读取,从文件中读取
fscanf(pf2, "%s %d %f", s1.name, &(s1.age), &(s1.score));
//写入到标准输出流中,屏幕上
fprintf(stdout, "%s %d %f\n", s1.name, s1.age, s1.score);
//关闭文件
fclose(pf2);
pf2 = NULL;
return 0;
}
sscanf/sprintf从字符串中格式化读取/输出数据
struct Stu
{
char name[20];
int age;
float score;
};
int main()
{
struct Stu s = {"zhangsan", 20, 66.5f};
char buf[200] = { 0 };
//sprintf可以把结构化的数据转换为一个字符串
sprintf(buf, "%s %d %f", s.name, s.age, s.score);
printf("按照字符串的形式:%s\n", buf);
struct Stu tmp = { 0 };
//sscanf可以把一个字符串转换为一个结构化的数据
sscanf(buf, "%s %d %f", tmp.name, &(tmp.age), &(tmp.score));
printf("按照格式化的形式:%s %d %f\n", tmp.name, tmp.age, tmp.score);
return 0;
}
七、文件结束判定
被错误使用的feof
牢记:在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。
而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
- 文本文件读取是否结束,判断返回值是否为EOF (fgetc),或者NULL(fgets)
例如:
fgetc判断是否为EOF.
fgets判断返回值是否为NULL.