文章目录
一、什么是文件
磁盘上的文件就是文件
但在程序设计中,我们一般谈的文件有两种:程序文件、数据文件(功能角度分类)
1.程序文件
包括源文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境后缀为.exe)
2.数据文件
文件的内容不一定是程序,而是程序 运行时读写的数据,比如程序运行需要从中读取数据的文件
或者输出内容的文件
本文主要以数据文件为中心进行简介
3.文件名
文件名包含三个部分
文件路径+文件主干+文件后缀
eg: C:\code\test.txt
路径: C:\code
文件名主干: test
文件后缀: .txt
二、文件的打开和关闭
1.文件指针
每个被使用的文件都在内存开辟了一个相应的文件信息区
用来存放文件的相关信息(文件名,文件状态,文件当前位置等).
这些信息是保存在一个结构体变量中的.
该结构体系统声明取名为 FILE
即 FILE 是一个结构体类型
具体内部实现如下(不同环境下,结构成员会有所差异))
struct _iobuf
{
//文件相关信息
//
//
};
typedef struct _iobuf FILE;
比如创建一个FILE*的指针变量
FILE * pf; //pf即文件指针变量
2.文件的打开和关闭
①fopen
#include<stdio.h>
.
FILE*fopen(const char* filename,const char* mode);
以mode的形式打开filename文件
其中mode主要分为以下几种
文件使用方式 | 含义 | 如果指定文件不存在 |
---|---|---|
“r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 |
“w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
“a”(追加) | 向文本文件尾添加数据 | 建立一个新的文件 |
“rb” (只读) | 为了输入数据,打开一个二进制文件 | 出错 |
“wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新的文件 |
“ab” (追加) | 向一个二进制文件尾添加数据 | 出错 |
“r+”(读写) | 为了读和写,打开一个文本文件 | 出错 |
“w+”(读写) | 为了读和写,建立一个新文件 | 建立一个新的文件 |
“a+”(读写) | 打开一个文件,在文件尾进行读写 | 建立一个新的文件 |
“rb+”(读写) | 为了读和写,打开一个二进制文件 | 出错 |
“wb+”(读写) | 为了读和写,建立一个二进制文件 | 建立一个新的文件 |
“ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 建立一个新的文件 |
以一段代码介绍fopen实际使用
FILE * pf = fopen("test.dat","r");//test.dat 位于该项目同一文件内可以被打开,或创建于同一文件
//但是并不代表只能打开同项目文件
FILE * pf = fopen("D:\\2023_code\\class\\test.dat","r");
//注意是\\而不是\
//给定一块指定的空间也可以打开或创建
if(pf == NULL)
{
perror("fopen"); //如果打开失败返回错误信息
return 1;
}
//处理文件
//关闭文件
fclose(pf);
pf = NULL;
②fclose
#include<stdio.h>
.
int fclose(FILE* stream);
三、文件的顺序读写
1.读写函数表
功能 | 函数名 | 适用于 |
---|---|---|
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输出流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出 | fwrite | 文件 |
2.“输入”,"输出"是什么
3.流
流是一个高度抽象的概念
同样的我们以图的形式展现流
具体理解可能略微生硬
在fputc中我们可以在逐渐深入了解
4.fputc
#include<stdio.h>
.
int fputc(int c,FILE*stream);
将一个字符c,输出到 stream(输出流) 中
举一个实例
函数运行后观察该项目文件夹
文件夹内创建了我们输入的文件名,并向文件内输出了我们想要的结果
当然输出流可以改变,比如stdout(标准输出流–屏幕)
5.fgetc
#include<stdio.h>
.
int fgetc(FILE * stream);
从流中读取单字符
每使用一次向后偏移一次
若读取失败则返回EOF( -1 )
举个例子
先将要读取的文件加入字符
运行程序后
同样的,输入流也可以不仅仅是文件,也可以是stdin(标准输入流—键盘)
输入后再打印
6.fputs
#include<stdio.h>
.
int fputs(const char * string, FILE * stream);
7.fgets
#include<stdio.h>
.
char * fgets(char * string , int n ,FILE * stream);
从流中读取最多n个字符到string
返回string的地址
举个例子
在文件中提前放入字符串
然后读取打印
可以看到打印成功
但是之所以只打印了三个字符
是因为需要预留一字节空间存储'\0'
8.fprintf
#include<stdio.h>
.
int fprintf(FILE * stream , const char * format [,argument]....);
struct S
{
char arr[10];
int num;
float sc;
};
int main()
{
struct S s = {"abcdef",10,5.5f};
//对格式化数据进行写文件
FILE*pf = fopen("test.dat","w");
if(pf == NULL)
{
perror("fopen");
return 1;
}
//写文件
fprintf(pf,"%s %d %f",s.arr,s.num,s.sc); //可以理解为fprintf为输出函数
//在pf文件内输出这几组数据
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
fprintf也可以适用于其他输出流
9.fscanf
#include<stdio.h>
.
int fscanf(FILE*stream, const char * format[,argument]....);
继上一段代码将结构中的数据存储于文件中
我们也可以读取文件中的结构
struct S
{
char arr[10];
int num;
float sc;
};
struct A
{
char arr[10];
int num;
float sc;
};
int main()
{
//对格式化数据进行读文件
FILE*pf = fopen("test.dat","r");
if(pf == NULL)
{
perror("fopen");
return 1;
}
//读文件
struct S s = {0};
fscanf(pf,"%s %d %f",s.arr,s.num,s.sc); //从pf中读取结构,再放到已之前输出的结构中
fscanf(pf,"%s %d %f",a.arr,a.num,a.sc); //同类型的结构不能放入,报错
//打印
printf("%s %d %f",s.arr,s.num,s.sc);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
也可使用于其他流
10.fwrite
#include<stdio.h>
.
size_t fwrite(const void*buffer,size_t isze,size_t count,FILE* stream);
该函数作用是
再stream流中写入大小为size,个数为count的二进制buffer数据
struct S
{
char arr[10];
int num;
float sc;
};
int main()
{
//对格式化数据进行写文件
FILE*pf = fopen("test.dat","w");
if(pf == NULL)
{
perror("fopen");
return 1;
}
//写文件
struct S s = { "abcde",10,5.5f };
fwrite(&s,sizeof(struct S),1,pf);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
二进制数据
引出fread函数
11.fread
#include<stdio.h>
.
size_t fread(void * buffer,size_t size,size_t count,FILE*stream);
该函数作用为
从stream流中读取count个大小为size的数据放入buffer内
我们以上述代码带入的结果测试一下
struct S
{
char arr[10];
int num;
float sc;
};
int main()
{
//对格式化数据进行读文件
FILE*pf = fopen("test.dat","r");
if(pf == NULL)
{
perror("fopen");
return 1;
}
//读文件
struct S s = {0};
fread(&s,sizeof(struct S),1,pf);
//打印观察结果
printf("%s %d %f",s.arr,s.num,s.sc);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
二进制数据可以被读取为我们可以读懂的数据
12.sscanf、sprintf
#include<stdio.h>
.
int sscanf(const char * buffer,const char*format[,argument]...);
从一段被格式化转换为字符串的数据
还原出原本的格式化数据
#include<stdio,h>
.
int sprintf(char*buffer,const char*format[,argumen]...);
把一个格式化的数据(包含各种类型)
转化为字符串存储至buffer
struct S
{
char arr[10];
int age;
float f;
};
int main()
{
struct S s = {"hello",20,5.5f};
char buf[100] = {0};
sprintf(buf,"%s %d %f",s.arr,s.age,s.f);
//将s的格式化数据全部转换为字符串存入到buf内
printf("%s\n",buf);
return 0;
}
打印结果如上
我们还可以将这段字符串通过fscanf还原出结构中的数据
struct S
{
char arr[10];
int age;
float f;
};
int main()
{
struct S s = {"hello",20,5.5f};
struct S tmp = {0};
char buf[100] = {0};
sprintf(buf,"%s %d %f",s.arr,s.age,s.f);
//将s的格式化数据全部转换为字符串存入到buf内
printf("%s\n",buf);
//从buf数据中还原出一个结构体数据
sscanf(buf, "%s %d %f",tmp.arr,&(tmp.age),&(tmp.f));
printf("%s %d %f\n",tmp.arr,tmp.age,tmp.f);
return 0;
}
四、文件的随机读写
在文件顺序读写中,我们发现以fgetc为例
总是从第一个字符开始,并向后逐一递增
怎样可以自定义读写方式呢
这里就要用到文件随机读写的知识
1.fseek
#include<stdio.h>
.
int fseek(FILE * stream,long int offset,int origin);
将指针定位至与起始位置 origin 偏移 offset 的位置
origin分为三种情况
SEEK_CUR — 当前文件指针的位置
SEEK_END — 文件末尾
SEEK_SET — 文件的起始位置
测试一下实际情况
int main()
{
FILE* pf = fopen("test.dat","r");
if(pf == NULL)
{
perror("fopen");
return 1;
}
//读取文件
int ch = fgetc(pf);
printf("%c\n",ch); //a
//调整文件指针
fseek(pf,1,SEEK_CUR); //将指针向回偏移1,从a到c
ch = fgetc(pf);
printf("%c\n",ch); //c
ch = fgetc(pf);
printf("%c\n",ch);//d
//关闭文件
fclose(pf);
pf=NULL;
return 0;
}
2.ftell
#include<stdio.h>
.
long int ftell(FILE * stream);
返回文件指针相对于起始位置的偏移量
int main()
{
FILE* pf = fopen("test.dat","r");
if(pf == NULL)
{
perror("fopen");
return 1;
}
//读取文件
int ch = fgetc(pf);
printf("%c\n",ch); //a
//调整文件指针
fseek(pf,1,SEEK_CUR); //将指针向回偏移1,从a到c
ch = fgetc(pf);
printf("%c\n",ch); //c
ch = fgetc(pf);
printf("%c\n",ch);//d
int ret = ftell(pf);
printf("%d",ret); //打印结果为4
//关闭文件
fclose(pf);
pf=NULL;
return 0;
}
3.rewind
#include<stdio.h>
.
void rewind(FILE*stream);
让文件指针回到起始的位置
还以上述举例
int main()
{
FILE* pf = fopen("test.dat", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//调整文件指针
fseek(pf, 3, SEEK_CUR); //将指针向回偏移3
int ch = fgetc(pf);
printf("%c\n", ch); //d
int ret = ftell(pf);
printf("%d\n", ret); //打印结果为4
rewind(pf); //使文件指针回到起始位置
ret = ftell(pf);
printf("%d\n", ret); //观察偏移量
ch = fgetc(pf);
printf("%c\n", ch);//a
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
五、文件读取结束的判定(feof)
feof的作用使
判断文件是读取失败结束的,还是遇到文件尾结束的
对于文本文件:
通过判定返回值来确定
判断返回值是否尾EOF(fgetc),或者NULL(fgets)
fgetc函数在读取结束后返回的是EOF
正常读取时返回的是读取到的字符的ASCII值
fgets函数在读取结束时返回的是NULL
正常读取时返回的是,存放字符串空间的起始地址
对于二进制文件:
判断返回值是否小于实际要读的个数
fread函数在读取的时候,返回的是实际读取到的完整元素的个数
如果发现读取到完整的元素个数小于 指定元素个数,这就是最后一次读取
举个例子
//文件拷贝
int main()
{
FILE * pread = fopen("test.txt","r");
if(pread == NULL)
{
perror("pread");
return 1;
}
FILE * pwrite = fopen("test.txt2","w");
if(pwrite == NULL)
{
//如果失败直接返回,文件没有关闭,会出问题
fclose(pread);
pread = NULL;
return 1;
}
//开始拷贝
int ch = 0;
while( ch = fgetc(fread) != EOF)
{
fputc(ch,fwrite);
}
//判断文件怎样结束
if(feof(pread)) //如果读取成功返回非0
{
printf("遇到文件结束标志,文件正常结束");
}
else if(ferror(pread)
{
printf("文件读取失败");
}
//关闭文件
fclose(pread);
pread = NULL;
fclose(pwrite);
pwrite = NULL;
}
六、文件缓冲区
简而言之就是
从程序到文件、从文件到程序
必须塞够足够的数据或者刷新( fflush (pf) )(高版本无法使用刷新)
数据才会完全输送到目的地
七、文本文件和二进制文件
1.数据在内存中以二进制的形式存储,如果不加以转换输出到外存,就是二进制文件
2.如果要求在外存以ASCII的形式存储,则需要在存储前转换
以ASCII字符的形式存储的文件就是文本文件
一个数据在内存中是怎么存储的呢
字符一律以ASCII形式存储
数值即可以ASCII存储也可以二进制形式存储
假设整数10000
ASCII存储需要5个字符 --五字节
而二进制只需要占4字节 --转换为二进制再转换为16进制计算