上上篇文章,我们介绍了文件和文件操作函数,现在我们来练习一下所学文件操作的相关函数吧!
实践出真知~
文件的打开和关闭
我们首先练习一下文件的打开和关闭:
每个被使⽤的⽂件都在内存中开辟了⼀个相应的⽂件信息区,⽤来存放⽂件的相关信息(如⽂件的名字,⽂件状态及⽂件当前的位置等)。这些信息是保存在⼀个结构体变量中的。该结构体类型是由系统声明的,取名FILE。
truct _iobuf { char *_ptr; int _cnt; char *_base; int _flag; int _file; int _charbuf; int _bufsiz; char *_tmpfname; }; typedef struct _iobuf FILE;
⼀般都是通过⼀个FILE的指针来维护这个FILE结构的变量,这样使⽤起来更加⽅便。
下⾯我们可以创建⼀个FILE*的指针变量:
FILE* pf;//⽂件指针变量
定义pf是⼀个指向FILE类型数据的指针变量。可以使pf指向某个⽂件的⽂件信息区(是⼀个结构体变量)。通过该⽂件信息区中的信息就能够访问该⽂件。也就是说,通过⽂件指针变量能够间接找到与它关联的⽂件。
如图所示,我们在文件里写入成功了。
# define _CRT_SECURE_NO_WARNINGS #include <stdio.h> int main() { FILE* pFile; //打开⽂件 pFile = fopen("project2", "w"); //⽂件操作 if (pFile != NULL) { fputs("fopen example", pFile); //关闭⽂件 fclose(pFile); } return 0; }
让我们打开这个文件,看看到底发生了什么吧。
接着让我们来到文件的随机读写
fseek
根据⽂件指针的位置和偏移量来定位⽂件指针。
int fseek ( FILE * stream, long int offset, int origin );
ftell
long int ftell ( FILE * stream );
#include <stdio.h> int main () { FILE * pFile; long size; pFile = fopen ("myfile.txt","rb"); if (pFile==NULL) perror ("Error opening file"); else { fseek (pFile, 0, SEEK_END); // non-portable size=ftell (pFile); fclose (pFile); printf ("Size of myfile.txt: %ld bytes.\n",size); } return 0; }
rewind
void rewind ( FILE * stream );
让文件指针回到起始位置
#include <stdio.h> int main () { int n; FILE * pFile; char buffer [27]; pFile = fopen ("myfile.txt","w+"); for ( n='A' ; n<='Z' ; n++) fputc ( n, pFile); rewind (pFile); fread (buffer,1,26,pFile); fclose (pFile); buffer[26]='\0'; printf(buffer); return 0; }
这段代码的主要功能是:
- 打开一个名为 "myfile.txt" 的文件,使用 "w+" 模式,这表示文件会被打开用于读写。如果文件不存在,它会被创建。如果文件已存在,它的内容会被清空。
- 使用 for 循环从字符 'A' 到 'Z',并将每个字符写入到文件中。
- 使用 rewind 函数将文件指针重新定位到文件的开头。
- 使用 fread 函数从文件中读取26个字节到 buffer 数组中。
- 关闭文件。
- 在 buffer 的第27个位置(索引为26)添加一个空字符(null terminator)\0,以确保 buffer 是一个合法的C字符串。
- 使用 printf 打印 buffer 的内容。
现在,让我们分析代码的结果:
- 当循环执行时,它会将字符 'A' 到 'Z' 写入文件。
- 使用 fread 读取时,会读取这26个字符。
- 但是,这里有一个问题。fread 读取的字节不会转换为字符串的终止符,因此在添加 \0 之前,buffer 并不是一个合法的C字符串。但是,在这段代码中,你确实在读取后添加了 \0,所以这不是一个错误。
最终,buffer
将包含字符 'A' 到 'Z',并且以 \0
结尾。因此,当使用 printf
打印 buffer
时,它将输出:
ABCDEFGHIJKLMNOPQRSTUVWXYZ |
文件结束的判定
文本文件:
#include <stdio.h> #include <stdlib.h> int main(void) { int c; // 注意:int,⾮char,要求处理EOF FILE* fp = fopen("test.txt", "r"); if(!fp) { perror("File opening failed"); return EXIT_FAILURE; } //fgetc 当读取失败的时候或者遇到⽂件结束的时候,都会返回EOF while ((c = fgetc(fp)) != EOF) // 标准C I/O读取⽂件循环 { putchar(c); } //判断是什么原因结束的 if (ferror(fp)) puts("I/O error when reading"); else if (feof(fp)) puts("End of file reached successfully"); fclose(fp); }
二进制文件:
#include <stdio.h> enum { SIZE = 5 }; int main(void) { double a[SIZE] = {1.,2.,3.,4.,5.}; FILE *fp = fopen("test.bin", "wb"); // 必须⽤⼆进制模式 fwrite(a, sizeof *a, SIZE, fp); // 写 double 的数组 fclose(fp); double b[SIZE]; fp = fopen("test.bin","rb"); size_t ret_code = fread(b, sizeof *b, SIZE, fp); // 读 double 的数组 if(ret_code == SIZE) { puts("Array read successfully, contents: "); for(int n = 0; n < SIZE; ++n) printf("%f ", b[n]); putchar('\n'); } else { // error handling if (feof(fp)) printf("Error reading test.bin: unexpected end of file\n"); else if (ferror(fp)) { perror("Error reading test.bin"); } } fclose(fp); }
文件缓冲区
#include <stdio.h> #include <windows.h> //VS2019 WIN11环境测试 int main() { FILE*pf = fopen("test.txt", "w"); fputs("abcdef", pf);//先将代码放在输出缓冲区 printf("睡眠10秒-已经写数据了,打开test.txt⽂件,发现⽂件没有内容\n"); Sleep(10000); printf("刷新缓冲区\n"); fflush(pf);//刷新缓冲区时,才将输出缓冲区的数据写到⽂件(磁盘) //注:fflush 在⾼版本的VS上不能使⽤了 printf("再睡眠10秒-此时,再次打开test.txt⽂件,⽂件有内容了\n"); Sleep(10000); fclose(pf); //注:fclose在关闭⽂件的时候,也会刷新缓冲区 pf = NULL; return 0; }