文件的打开和关闭
在编写程序的时候,在打开文件的同时,都会返回一个FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系。
fopen函数和fclose函数
FILE * fopen ( const char * filename, const char * mode );
int fclose ( FILE * stream );
打开方式
文件使用方式 | 含义 | 如果指定文件不存在 |
---|---|---|
“r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 |
“w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
“w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
“a”(追加) | 向文本文件尾添加数据 | 出错 |
“rb”(只读) | 为了输入数据,打开一个二进制文件 | 出错 |
“wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新的文件 |
“ab”(追加) | 向一个二进制文件尾添加数据 | 出错 |
“r+”(读写) | 为了读和写,打开一个文本文件 | 出错 |
“w+”(读写) | 为了读和写,建议一个新的文件 | 建立一个新的文件 |
“a+”(读写) | 打开一个文件,在文件尾进行读写 | 建立一个新的文件 |
“rb+”(读写) | 为了读和写打开一个二进制文件 | 出错 |
“wb+”(读写) | 为了读和写,新建一个新的二进制文件 | 建立一个新的文件 |
“ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 建立一个新的文件 |
下面我们来举个例子
int main()
{
//fopen函数是以写的形式打开的
//如果文件不存在,会创建文件
//fopen函数如果是以读的形式打开
//文件不存在打开失败
FILE* pf = fopen("data.text", "w");
if (pf == NULL)
{
printf("打开文件失败\n");
printf("%s\n", strerror(errno));
return 1;//失败返回
}
//打开文件成功
printf("打开文件成功\n");
//读写文件
int i;
for (i = 'a';i <= 'z';i++)
{
fputc(i, pf);
}
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
文件的顺序读写
功能 | 函数名 | 适用于 |
---|---|---|
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输出流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出 | fwrite | 文件 |
文件的随机读写
fseek
根据文件指针的位置和偏移量来定位文件指针
int fseek(FILE* stream, long int offset, int origin);
ftell
返回文件指针相对于起始位置的偏移量
long int ftell ( FILE * stream );
rewind
让文件指针的位置回到文件的起始位置
void rewind ( FILE* stream );
让我们来看个例子
int main()
{
FILE* pf = fopen("data.txt", "r");
if (pf == NULL)
{
return 1;
}
//读取
int ch = fgetc(pf);
printf("%c\n", ch);
//定位文件指针到文件的起始位置
//fseek(pf,-2,SEEK-CUR);
//fseek(pf, 0, SEEK_SET);
//printf("%d\n", ftell(pf));
rewind(pf);
ch = fgetc(pf);//要在这里读取'a'
printf("%c\n", ch);
fclose(pf);
pf = NULL;
return 0;
}
文件结束判定
被错误使用的 feof
在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束,而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
1、文本文件读取是否结束,判断返回值是否为EOF (fgetc),或者NULL(fgets)
2、二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
int main()
{
FILE* pf = fopen("data.txt", "r");
if (pf == NULL)
{
return 1;
}
//读取
int ch = 0;
while (ch = fgetc(pf) != EOF)
{
printf("%c ", ch);
}
//找结束的原因
if (ferror(pf))
{
printf("读取是发生错误而失败\n");
}
else if (feof(pf))
{
printf("遇到文件末尾,而结束\n");
}
fclose(pf);
pf = NULL;
return 0;
}
文件缓冲区
ANSIC 标准采用“缓冲文件系统”处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。
int main()
{
FILE* pf = fopen("tdst.txt", "w");
fputs("abccdef", pf);//先将代码放在输出缓冲区
printf("睡眠10秒-已经写数据了,打开文件,发现没有内容\n");
sleep(10000);
printf("刷新缓冲区\n");
fflush(pf);//刷新缓冲区时,才将输出缓冲区的数据写到文件(磁盘)
//注:fflush在高版本的vs上不能使用了
printf("在睡眠10s-此时,再次打开文件,文件有内容了\n");
Sleep(10000);
fclose(pf);
//注:fclose再关闭文件时,也会刷新缓冲区
pf = NULL;
return 0;
}