文件操作
一、文件分类
1、为什么使用文件?
在使用VS书写代码的时候,在每一次运行后,代码里的数据都会被清空,好像没有存在过一样,这是因为代码是在内存上运行的,其数据相关的内容也会在内存上暂时存放,如果需要将数据保存下来,就需要储存到硬盘中,这里我们就可以运用到“文件”了,我们通过打开,读写等方式对文件进行操作,就可以将数据内容存储到文件中,即使程序运行完毕,我们的数据也不会消失
1. int main()
2. {
3. //打开文件
4. FILE*pf = fopen("test.txt", "w");
5. if(pf == NULL)
6. {
7. perror("fopen");
8. return 1;
9. }
10. //写文件
11. fputs("holle word", pf);
12. //关闭文件
13. fclose(pf);
14. pf = NULL;
15. return 0;
16. }
运行了这段代码后,在本次代码的目录下会生成一个 “ test.txt ” 文件,并将 " holle word " 写入文件中,在代码运行完毕后,这段文字也不会消失。
下面我们具体来了解C语言中文件操作相关的内容。
2、什么是文件?
磁盘(硬盘)上的⽂件是⽂件。
但是在程序设计中,我们⼀般谈的⽂件有两种:程序⽂件、数据⽂件(从⽂件功能的⻆度来分类的)。
(1)、程序文件
程序⽂件包括源程序⽂件(后缀为.c),⽬标⽂件(windows环境后缀为.obj),可执⾏程序(windows环境后缀为.exe)。
(2)、数据文件
⽂件的内容不⼀定是程序,⽽是程序运⾏时读写的数据,⽐如程序运⾏需要从中读取数据的⽂件,或者输出内容的⽂件。
3、二进制文件和文本文件
在我们C语言中,文件分为二进制文件和文本文件。
文本文件:在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的⽂件就是⽂本⽂件。
二进制文件:数据在内存中以⼆进制的形式存储,如果不加转换的输出到外存的⽂件中,就是⼆进制⽂件。
(1)、文本文件
我们将数据以ASCII码的形式储存在文件中,此时我们的文件就可以称之为文本文件。
用通俗的大白话来说就是,文本文件我们是可以看得懂的。
例如:
(2)、二进制文件
而二进制的文件,是将数据以二进制的形式存储到文件中,这时的文件即使我们打开了,也会发现是一段根本看不懂的乱码,这是因为二进制文件是给机器看的,而不是面向用户的。
(3)、两者区别
在上文中我们看到的两种文件类型,那可能会有未来的资深程序猿问了,写代码就是为了给用户看的,那为何不直接全部以ASCII码的形式存储,这样都能看得懂呢?
这就不得不引入一个例子了。
例如:我们要把数字5存储到文件中。
我们能看到存储5的时候如果用ASCII的形式存储只会使用1个字节的内存,但如果使用二进制存储却会使用四个字节,确实,ASCII的形式会节省空间。
那我们如果要存储10000到文件中呢?
我们发现如果将10000以ASCII的形式存储到文件中用了五个字节的内存空间,反而比二进制的方式更多。
所以,这两种存储数据的方式并无好坏之分,区别就在于存储的数据是什么样的,如果我们能在存储数据的时候选择更加高效的存储方式,那便会提高我们的代码效率。
二、文件的打开和关闭
在了解文件的打开和关闭时,我们需要先了解一个概念——流。
作为一个高度抽象的概念插入到我们的C语言学习中,我们可以将流理解为一个传送带,我们需要存放数据时,将数据内容放到传送带(即流)上,存储的目标就会从这条传送带上拿取数据。在获取数据时也是一样的道理。
在C语言中,我们在输入输出数据时,并不需要手动的打开流,因为在程序启动时,我们的编译器就会默认的打开三个流。
即
stdin - 标准输⼊流,在⼤多数的环境中从键盘输⼊,scanf函数就是从标准输⼊流中读取数据。
stdout - 标准输出流,⼤多数的环境中输出⾄显⽰器界⾯,printf函数就是将信息输出到标准输出流中。
stderr - 标准错误流,⼤多数环境中输出到显⽰器界⾯。
而拥有了这三个流,我们就可以将数据通过 scanf 和 printf 进行操作了。
1、文件指针
文件在C语言中又是如何定义的呢,其实在我们的头文件 stdio.h 中就有文件的定义。
1.struct _iobuf {
2. char *_ptr;
3. int _cnt;
4. char *_base;
5. int _flag;
6. int _file;
7. int _charbuf;
8. int _bufsiz;
9. char *_tmpfname;
10.};
11.typedef struct _iobuf FILE;
在头文件 stdio.h 中会有一个结构体类型的声明,在结构体中存放了这个文件的各种信息,这里我们先不做了解,但可以看到最后,将文件类型重新定义为了FILE。
所以我们在平时的使用中,就可以通过FILE来创建指针,代表着文件的地址,进而对文件进行一系列的操作,
2、文件的打开和关闭
在对文件进行操作的时候,我们需要先将文件打开,在操作完成后,也需要对文件进行关闭,并将代表文件地址的指针置为空指针,避免其成为野指针,
我们通过fopen来打开文件,通过fclose来关闭文件。
1. int main()
2. {
3. FILE*PF = fopen("test.txt", "w");
4. //需要打开的文件 //打开后要进行的操作
5. fclose(pf);//在关闭文件只需要将存放文件地址的指针传入即可
6. //代表文件地址的指针
7. pf = NULL;//关闭后将指针置为空指针,是一个不错的习惯
8. return 0;
9. }
3、文件的打开模式
上面是打开文件进行操作的方式,每一次打开文件只能有一种操作方式。
值得注意的是,如果我们在读文件的时候,打开的文件并不存在,那么代码就会报错从而停止运行。
但如果是写文件的时候,如果文件不存在,那么就会在根目录下创建一个新的文件,并进行操作,但如果使用“w” 或 “wb” 进行写操作时,如果原文件中存在数据内容,那么这个操作将会将原文件中的内容清空,并进行新一轮的写操作。
其余的文件使用方式大家可以对照上表进行了解。
4、文件的顺序读写函数介绍
下面说的适⽤于所有输⼊流⼀般指适⽤于标准输⼊流和其他输⼊流(如⽂件输⼊流);所有输出流⼀般指适⽤于标准输出流和其他输出流(如⽂件输出流)。
让我们一个一个来进行比较:
4.1 、fputc
1. int main()
2. {
3. FILE* pf = fopen("test.txt", "w");
4. if(pf == NULL)
5. {
6. perror("fopen");
7. return 0;
8. }
9.
10. fputc('a', pf);
11.
12. fclose(pf);
13. pf = NULL;
14. return 0;
15. }
我们通过将需要写入的字符和代表文件地址的指针p传给fputc函数,即可将字符输入到文件中。
4.2 、fgetc
我们可以用fgetc函数再将刚刚存入test.txt的内容读出来
1. int main()
2. {
3. FILE* pf = fopen("test.txt", "r");
4. if(pf == NULL)
5. {
6. perror("fopen");
7. return 0;
8. }
9.
10. char n = fgetc(pf);
11. printf("%c\n", n);
12.
13. fclose(pf);
14. pf = NULL;
14. return 0;
15. }
4.3 、fputs
相对于fgetc,fgers可以一次性输入一个字符串。
1. int main()
2. {
3. FILE* pf = fopen("test.txt", "w");
4. if(pf == NULL)
5. {
6. perror("fopen");
7. return 0;
8. }
9.
10. fputs('holle word', pf);
11.
12. fclose(pf);
13. pf = NULL;
14. return 0;
15. }
4.3 、fgets
fgets可以将存入fputs中的数据打印出来
1. int main()
2. {
3. FILE* pf = fopen("test.txt", "r");
4. if(pf == NULL)
5. {
6. perror("fopen");
7. return 0;
8. }
9.
10. char arr[15] = { 0 };
11. fgets(arr,15, pf);
12. printf("%s\n", arr);
13.
14. fclose(pf);
15. pf = NULL;
14. return 0;
15. }
4.5 、fprintf
int main()
{
FILE* p = fopen("text.txt", "w");
if (p == NULL)
{
perror("fopen");
return 1;
}
fprintf(p, "%d %d %s", 100, 200, "abcd");
fclose(p);
p = NULL;
return 0;
}
此时文件中的 100 和 200 已经不再是整型了,而是六个字符型。
4.6 、fscanf
上文中我们将 100 和 200 以及 abcd存储到文件中,但文件中的数据都是字符型的,我们可以将其通过fscanf函数将其以特定格式存储到指定的参数中。例如
int main()
{
FILE* p = fopen("text.txt", "r");
if (p == NULL)
{
perror("fopen");
return 1;
}
int a = 0;
int b = 0;
char arr[10] = { 0 };
fscanf(p, "%d %d %s", &a, &b, arr);
printf("%d %d %s\n", a, b, arr);
fclose(p);
p = NULL;
return 0;
}
4.7 、fwrite
此函数为二进制写入函数,故写入的数据会转换为二进制存储到文件中,我们可以通过fread来读取二进制文件内容。
int main()
{
FILE* p = fopen("text.txt", "wb");
if (p == NULL)
{
perror("fopen");
return 1;
}
int arr[5] = { 1,2,3,4,5 };
fwrite(arr, sizeof(int), sizeof(arr), p);
fclose(p);
p = NULL;
return 0;
}
### 4.8 、fread
这个函数就可以帮助我们将上一个函数存储的信息进行读操作了,并且我们可以将它打印出来。
int main()
{
FILE* p = fopen("text.txt", "rb");
if (p == NULL)
{
perror("fopen");
return 1;
}
int arr[5] = { 0 };
fread(arr, sizeof(int), sizeof(arr), p);
for (int i = 0; i < 5; i++)
{
printf("%d ", arr[i]);
}
fclose(p);
p = NULL;
return 0;
}
到此我们就将文件操作相关的基本函数了解了大概,但还需给看官补充一些有关文件的随机读写的函数。
三、文件的随机读写
上文中的文件操作函数,只能按照顺序对文件内容进行读写,那我们如果想要制定一个位置来读写呢?
例如:文件中存放了字符串abcde,如果按照顺序读写的话在读取a后只能读取b,那我们如何在读取到a后读取e呢?这就用到了我们下面讲的文件随机读写函数。
1、fseek
对于此函数,我们分别传文件的地址和对于指定位置的偏移量(offset)以及指定位置(origin)三个参数。这就要引入下表中的信息。
指定位置名 | 含义 |
---|---|
SEEK_CUR | 文件指针当前位置 |
SEEK_END | 文件指针的结尾 |
SEEK_SET | 文件指针的开头 |
那具体怎么操作呢?
int main()
{
FILE* p = fopen("example.txt", "wb");
if (p == NULL)
{
perror("fopen");
return 1;
}
fputs("This is an apple.", p);
fseek(p, 9, SEEK_SET);
fputs(" sam", p);
fclose(p);
p = NULL;
return 0;
}
我们可以看到原本p中存放了 This is an apple. 的字符串,在我们通过使用fseek函数后,将文件指针位置以文件起始位置为起点,偏移量为9进行移动,从而得到了“ n apple ”的首地址,再将 sam 存入其中,故最后文件中存放的数据就是 This is a sample 的字符串。
2、ftell
这个函数的功能就比较简单了,我们在存放完数据后,将p指针传给函数ftell,其返回值就代表着此时文件指针相对于文件开头的偏移量了。
int main()
{
FILE* p = fopen("example.txt", "wb");
if (p == NULL)
{
perror("fopen");
return 1;
}
fputs("This is an apple.", p);
int n = 0;
n = ftell(p);
printf("%d\n", n);
fclose(p);
p = NULL;
return 0;
}
3、rewind
此函数可以让文件指针无论身处何处,都能重置为文件的开头位置。
int main()
{
FILE* p = fopen("example.txt", "w");
if (p == NULL)
{
perror("fopen");
return 1;
}
fputs("Lili love C!", p);
rewind(p);
fputs("Stiv", p);
fclose(p);
p = NULL;
return 0;
}
代码中原本是Lili love C!,在我们完成第一次写入时,使用rewind让文件指针回到起始位置,在进行Stiv的写入。最终,文件中存储的内容就是 Stiv love C!了。
四、文件读取结束的判定
1、被错误使用的feof
牢记:在⽂件读取过程中,不能⽤feof函数的返回值直接来判断⽂件的是否结束。
feof 的作⽤是:当⽂件读取结束的时候,判断是读取结束的原因是否是:遇到⽂件尾结束
1. ⽂本⽂件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
例如:
fgetc 判断是否为 EOF .
fgets 判断返回值是否为 NULL .
2. ⼆进制⽂件的读取结束判断,判断返回值是否⼩于实际要读的个数。
例如: fread判断返回值是否⼩于实际要读的个数。
#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);
}
五、结语
十分感谢您观看我的原创文章。
本文主要用于个人学习和知识分享,学习路漫漫,如有错误,感谢指正。
如需引用,注明地址。