1.为什么要使用文件
在生活中,我们在电脑或手机上下载了一些东西,当我们关机再开机后,那些东西依然存在,只有当我们主动删除它的时候,它才会消失,使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。
2.文件的打开和关闭及读写
2.1文件指针
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息存储在结构体变量中,系统会将结构体改名为FILE,此后,我们只需要通过一个FILE类型的指针就可以访问相应的文件。
2.2打开文件
打开文件用的是fopen
函数,在程序结束时需要关闭!
FILE *fopen( const char *filename, const char *mode );
名称 | 详情 |
---|---|
参数 const char *filename | 文件路径 |
参数 const char *mode | 文件模式(读、写等等) |
返回类型 | FILE *(当操作失败时会返回NULL) |
2.3关闭文件
在程序结束时,需要用fclose
函数去关闭文件。
int fclose( FILE *stream );
名称 | 详情 |
---|---|
参数FILE *stream | 待关闭的文件指针 |
返回类型 | int(返回-1时,表示关闭失败) |
下面我们来看一个例子:
int main()
{
//打开文件
FILE* pf = fopen("C:\\Users\\10371\\Desktop\\haha.txt", "w+");
if (pf == NULL)
{
perror("fopen");
return -1;
}
//读文件
//
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
当我们以‘w+’的方式去操作文件时,在指定路径上没有该文件时,系统会创建该文件。如果我们想往刚刚创建的文件中写入东西的话该怎么办呢?我们来看下面的例子:
int main()
{
//打开文件
FILE* pf = fopen("C:\\Users\\10371\\Desktop\\haha.txt", "w+");
if (pf == NULL)
{
perror("fopen");
return -1;
}
//读文件
//输出字符
fputc('a', pf);
//将文件指针重新定位到文件的开头
rewind(pf);
//输入字符
char ch = 0;
ch = fgetc(pf);
printf("%c ", ch);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
输出结果如下:
a
D:\code\c-language-learning\test_10_4\Debug\test_10_4.exe (进程 25664)已退出,代码为 0。
按任意键关闭此窗口. . .
当然,我们还可以以其他的方式对文件进行输入输出(二进制、字符串等等)。
3.文件结束判断
C语言中的feof
函数,可用来判断是否位于文件末尾,参数为文件指针,如果不是文件末尾则返回0,否则返回非0值。
int feof( FILE *stream );
通常,我们还需要配合feeror
函数用来判断文件是否是异常结束,如果正常返回0,否则返回非0值。
int ferror( FILE *stream );
1.判断文本文件读取是否正常结束:
int main(void) {
int c; // 注意:int,非char,要求处理EOF
FILE* fp = fopen("C:\\Users\\10371\\Desktop\\haha.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);
return 0;
}
2.判断二进制文件读取是否正常结束:
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);
return 0;
}