前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >【C语言】文件操作详解

【C语言】文件操作详解

作者头像
大耳朵土土垚
发布于 2024-03-13 10:38:38
发布于 2024-03-13 10:38:38
15600
代码可运行
举报
文章被收录于专栏:c/c++c/c++
运行总次数:0
代码可运行

🥳🥳 1. 为什么使用文件?

如果没有⽂件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等再次运⾏程序,是看不到上次程序的数据的,如果要将数据进⾏持久化的保存,我们可以使⽤⽂件。

简单来说就是将需要用的数据保存在电脑中方便下次使用

✨✨2. 什么是文件?

磁盘上的⽂件是⽂件。 但是在程序设计中,我们⼀般谈的⽂件有两种:程序⽂件、数据⽂件(从⽂件功能的⻆度来分类的)。

2.1 程序⽂件

程序⽂件包括源程序⽂件(后缀为.c),⽬标⽂件(windows环境后缀为.obj),可执⾏程序(windows环境后缀为.exe)。 💫💫大家在写程序时就会发现多了很多.c .obj 等相关文件这些都是程序文件

2.2 数据文件

⽂件的内容不⼀定是程序,⽽是程序运⾏时读写的数据,⽐如程序运⾏需要从中读取数据的⽂件,或者输出内容的⽂件。

2.3 文件名

⼀个⽂件要有⼀个唯⼀的⽂件标识,以便用户识别和引用。 🥳🥳⽂件名包含3部分:⽂件路径+⽂件名主⼲+⽂件后缀 例如:c:\code\test.txt 其中test是文件名主干,.txt是文件后缀 为了⽅便起⻅,⽂件标识常被称为文件名

🥰🥰3. 二进制文件和文本文件?

根据数据的组织形式,数据⽂件被称为**文本文件或者二进制文件**。

数据在内存中以⼆进制的形式存储,如果不加转换的输出到外存,就是⼆进制⽂件。 如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的⽂件就是⽂本⽂件。

测试代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <stdio.h>
int main()
{
	int a = 10000;
	FILE* pf = fopen("test.txt", "wb");
	fwrite(&a, 4, 1, pf);//⼆进制的形式写到⽂件中

	fclose(pf);
	pf = NULL;
	return 0;
}

运行代码可以发现多了一个test.txt文件:

我们以二进制形式打开:

结果如下:

我们发现fwrite函数已经将a的值以二进制形式写入文件当中啦🥳🥳

🎉🎉4. 文件的打开和关闭

4.1 流和标准流

4.1.1 流

我们程序的数据需要输出到各种外部设备,也需要从外部设备获取数据,不同的外部设备的输⼊输出操作各不相同,为了⽅便程序员对各种设备进⾏⽅便的操作,我们抽象出了流的概念,我们可以把流 想象成流淌着字符的河。 ✨✨ C程序针对⽂件、画⾯、键盘等的数据输⼊输出操作都是通过流操作的。

4.1.2 标准流

那为什么我们从键盘输⼊数据,向屏幕上输出数据,并没有打开流呢?

那是因为C语⾔程序在启动的时候,默认打开了3个流: • stdin-标准输⼊流,在⼤多数的环境中从键盘输⼊,scanf函数就是从标准输⼊流中读取数据。 • stdout-标准输出流,⼤多数的环境中输出⾄显⽰器界⾯,printf函数就是将信息输出到标准输出 流中。 • stderr-标准错误流,⼤多数环境中输出到显⽰器界⾯。 这是默认打开了这三个流,我们使⽤scanf、printf等函数就可以直接进⾏输⼊输出操作的。 stdin、stdout、stderr三个流的类型是:FILE* ,通常称为⽂件指针。 C语⾔中,就是通过 FILE* ,通常称为⽂件指针。 FILE* 的⽂件指针来维护流的各种操作的。

4.2 ⽂件指针

缓冲⽂件系统中,关键的概念是“⽂件类型指针”,简称“⽂件指针”。 每个被使⽤的⽂件都在内存中开辟了⼀个相应的⽂件信息区,⽤来存放⽂件的相关信息(如⽂件的名字,⽂件状态及⽂件当前的位置等)。这些信息是保存在⼀个结构体变量中的。 该结构体类型是由系 统声明的,取名FILE.

例如,VS2013编译环境提供的stdio.h 头⽂件中有以下的⽂件类型申明:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 struct _iobuf {
 char *_ptr;
 int   _cnt;
 char *_base;
 int   _flag;
 int   _file;
 int   _charbuf;
 int   _bufsiz;
 char *_tmpfname;
 };
 typedef struct _iobuf FILE;

🥳🥳不同的C编译器的FILE类型包含的内容不完全相同,但是⼤同⼩异。 每当打开⼀个⽂件的时候,系统会根据⽂件的情况⾃动创建⼀个FILE结构的变量,并填充其中的信息,使⽤者不必关⼼细节。⼀般都是通过⼀个FILE的指针来维护这个FILE结构的变量,这样使⽤起来更加⽅便。

下⾯我们可以创建⼀个FILE*的指针变量:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 FILE* pf;//⽂件指针变量

✨✨定义pf是⼀个指向FILE类型数据的指针变量。可以使pf指向某个⽂件的⽂件信息区(是⼀个结构体变量)。通过该⽂件信息区中的信息就能够访问该⽂件。 也就是说,通过⽂件指针变量能够间接找到与它关联的⽂件。

4.3 ⽂件的打开和关闭

⽂件在读写之前应该先打开⽂件,在使⽤结束之后应该关闭⽂件。 在编写程序的时候,在打开⽂件的同时,都会返回⼀个FILE*的指针变量指向该⽂件,也相当于建⽴了 指针和⽂件的关系。 ✨✨ ANSIC规定使⽤ fopen 函数来打开⽂件, fclose 来关闭⽂件。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 //打开⽂件
FILE * fopen ( const char * filename, const char * mode );

 //关闭⽂件
int fclose ( FILE * stream );

mode表⽰⽂件的打开模式,下⾯都是⽂件的打开模式:

实例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/* fopen fclose example */
#include <stdio.h>
int main()
{
	FILE* pFile;
	//打开文件
	pFile = fopen("myfile.txt", "w");
	//文件操作
	if (pFile != NULL)
	{
		fputs("fopen example", pFile);
		//关闭文件
		fclose(pFile);
	}
	return 0;
}

💫💫5. 文件的顺序读写

5.1 顺序读写函数介绍

💥💥6. 文件的随机读写

6.1 fseek

根据文件指针的位置和偏移量来定位文件指针。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
int fseek ( FILE * stream, long int offset, int origin );

以下是对该函数参数的解读:

stream是你想要打开的文件指针; offset指的是你想要从哪开始读文件,相对于orgin也就是开始文件指针位置的偏移量大小,以字节为单位; origin表示指定文件指针从哪开始偏移 。 它有三个参数: SEEK_SET:表示从文件头开始偏移offset个字节 SEEK_CUR:表示从当前文件指针位置开始偏移offset个字节 SEEK_END:表示从文件尾部位置开始偏移offset个字节

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 /* fseek example */
 #include <stdio.h>
 int main ()
 {
  FILE * pFile;
  pFile = fopen ( "example.txt" , "wb" );
  fputs ( "This is an apple." , pFile );
  fseek ( pFile , 9 , SEEK_SET );//从头开始偏移9个字节
  fputs ( " sam" , pFile );
  fclose ( pFile );
  return 0;
 }

6.2 ftell

返回文件指针相对于起始位置的偏移量

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
long int ftell ( FILE * stream );

实例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/* ftell example : getting size of a file */
 #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);   
  }
 return 0;
 }

6.3 rewind

让文件指针的位置回到文件的起始位置

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
void rewind ( FILE * stream );

实例代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/* rewind example */
 #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';
 puts (buffer);
 return 0;
 }

🤩🤩7. 文件读取结束的判定

7.1 被错误使用的feof

牢记:在文件读取过程中,不能用feof函数的返回值直接来判断文件的是否结束。 feof 的作用是:当文件读取结束的时候,判断是读取结束的原因是否是:遇到文件尾结束。

  1. 文本文件读取是否结束,判断返回值是否为 EOF ( 例如: fgetc 判断是否为 EOF . fgets 判断返回值是否为 NULL ),
  2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。 例如: fread判断返回值是否小于实际要读的个数。

实例代码: 文本文件:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#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);
 }

二进制文件:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#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);
 }

🧨🧨8. 文件缓冲区

ANSIC 标准采用“缓冲文件系统”处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装 满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <stdio.h>
 #include <windows.h>
 //VS2013 WIN10环境测试
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}

这里可以得出一个结论: 因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区或者在文件操作结束的时候关闭文件。 如果不做,可能导致读写文件的问题。

🌹🌹9.结语

以上就是有关c语言文件操作的知识啦~大家都学废了吗,完结撒花 ~🥳🥳🎉🎉🎉

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-03-11,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
C语言——P/文件操作
如果没有文件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等再次运行程序,是看不到上次程序的数据的,如果要将数据进行持久化的保存,我们可以使用文件。
用户11015888
2024/03/11
2200
C语言——P/文件操作
C语言中的文件和文件操作
硬盘上的文件是文件。但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件(从文件功能角度来分类的)。
用户11305458
2024/10/09
1650
C语言中的文件和文件操作
深入挖掘C语言 ---- 文件操作
我们程序的数据需要输出到各种外部设备, 也需要从外部设备获取数据, 不同的外部设备的输入输出操作各不相同, 为了方便程序员对各种外部设备进行方便的操作, 我们抽象出了流的概念, 我们可以把流想象成流淌着字符的河.
用户11317877
2024/10/16
1390
深入挖掘C语言 ---- 文件操作
【C语言篇】文件操作(下篇)
上面的四个都是针对字符的输入输出,但是实际文件会有不同的数据类型,这时就需要用到格式化输入输出函数了
半截诗
2024/10/09
890
【C语言篇】文件操作(下篇)
轻松拿捏C语言——【文件操作】
程序文件:包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows 环境后缀为.exe)
用户11162265
2024/06/14
1100
轻松拿捏C语言——【文件操作】
C语言文件操作超详解
如果没有文件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等再次运行程序,是看不到上次程序的数据的,如果要将数据进行持久化的保存,我们可以使用文件。
fhvyxyci
2024/09/24
1410
C语言文件操作超详解
【C语言】文件操作
磁盘(硬盘)上的文件是文件。 但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件(从文件功能的角度来分类的)。
用户11290673
2024/09/25
1580
【C语言】文件操作
C语言文件操作
我们前面学习结构体时,写通讯录的程序,当通讯录运行起来的时候,可以给通讯录中增加、删除数据,此时数据是存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录程序的时候,数据又得重新录入,如果使用这样的通讯录就很难受。
绝活蛋炒饭
2024/12/16
1440
C语言文件操作
C语言——文件操作
磁盘上的文件是文件。 但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件
小李很执着
2024/06/15
2440
C语言——文件操作
C语言进阶-文件操作超详解
scanf/printf、fscanf/fprintf、sscanf/sprintf函数对比
用户9645905
2022/11/30
1K0
C语言进阶-文件操作超详解
文件操作(几乎最全)
我们写的程序的数据是存储在电脑的内存中,如果没有文件,程序退出,内存回收,数据就丢失了,等再次运行程序,是看不到上次程序的数据的,如果要将数据进行持久化的保存,我们可以使用文件。
Crossoads
2024/10/21
1280
文件操作(几乎最全)
抽丝剥茧C语言(高阶)文件操作+练习
我们前面了解结构体时,写了通讯录的程序,当通讯录运行起来的时候,可以给通讯录中增加、删除数据,此时数据是存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录程序的时候,数据又得重新录入,如果使用这样的通讯录就很难受。(前面我已经把通讯录完善了) 我们在想既然是通讯录就应该把信息记录下来,只有我们自己选择删除数据的时候,数据才不复存在。 这就涉及到了数据持久化的问题,我们一般数据持久化的方法有,把数据存放在磁盘文件、存放到数据库等方式。 使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。
有礼貌的灰绅士
2023/03/28
5480
抽丝剥茧C语言(高阶)文件操作+练习
文件操作——C语言
如果没有文件,我们写的程序的数据都是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等程序再次运行,我们是无法看到上次程序的数据的。如果想要将数据持久化的保存,我们可以使用文件来存储数据。
星辰与你
2024/10/17
1560
文件操作——C语言
C语言重点突破(六)文件操作
我们在前面的文章介绍了通讯录的程序,当通讯录运行起来的时候,可以给通讯录中增加、删除数据,此时数据是存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录程序的时候,数据又得重新录入,如果使用这样的通讯录就很难受。 我们在想既然是通讯录就应该把信息记录下来,只有我们自己选择删除数据的时候,数据才不复存在。 这就涉及到了数据持久化的问题,我们一般数据持久化的方法有,把数据存放在磁盘文件、存放到数据库等方式。 使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。
对编程一片赤诚的小吴
2024/01/23
1530
C语言重点突破(六)文件操作
C语言----文件操作
如果没有⽂件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等再次运⾏程序,是看不到上次程序的数据的,如果要将数据进⾏持久化的保存,我们可以使⽤⽂件。
Undoom
2024/09/23
1720
C语言----文件操作
文件操作学不懂,小代老师带你深入理解文件操作(下卷)
牢记:在⽂件读取过程中,不能⽤feof函数的返回值直接来判断⽂件的是否结束。 feof 的作⽤是:当⽂件读取结束的时候,判断是读取结束的原因是否是:遇到⽂件尾结束。
用户11319080
2024/10/17
810
文件操作学不懂,小代老师带你深入理解文件操作(下卷)
深度解析C语言文件操作
我们前面学习结构体时,写了通讯录的程序,当通讯录运行起来的时候,可以在通讯录中添加、删除数据,此时数据是存放在内存中的,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录时,数据又得重新导入,如果这样使用通讯录就很难受。 既然是通讯录就应该记录下来,只有我们自己选择删除数据得时候,数据才不复存在。这就涉及到了数据得持久化得问题,我们一般数据持久化得方法有,把数据存放在磁盘文件、存放到数据库等方式。 使用文件我们就可以将数据存放在电脑的硬盘上,做到数据的持久化。
Yui_
2024/10/16
1600
深度解析C语言文件操作
C语言:文件操作
如果没有⽂件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等再次运⾏程序,是看不到上次程序的数据的,如果要将数据进⾏持久化的保存,我们可以使⽤⽂件,因为文件是存放在硬盘上的
逆向-落叶
2024/10/28
2580
C语言:文件操作
C语言从入门到实战——文件操作
C语言中的文件操作是通过使用文件指针来实现的。可以使用标准库中的函数来打开、读取、写入和关闭文件。
鲜于言悠
2024/03/20
5100
C语言从入门到实战——文件操作
练习所学文件操作的相关函数
上上篇文章,我们介绍了文件和文件操作函数,现在我们来练习一下所学文件操作的相关函数吧!
用户11039545
2024/04/02
1090
练习所学文件操作的相关函数
相关推荐
C语言——P/文件操作
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验