当前位置: 首页 > news >正文

C语言_文件操作

1. 为什么使用文件 ?

我们写的程序的数据是存储在电脑内存中的,如果程序退出,内存就会回收,数据就会丢失;如果我们不想丢失数据,想对数据进行持久化的保存,就要用到文件了

2. 什么是文件 ?

磁盘(硬盘)上的文件就是文件

在程序设计中,文件从文件功能上共分为俩种:程序文件、数据文件

程序文件:程序文件包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境后缀为.exe)

数据文件:文件中是程序运行时读写的数据。有时我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使用,这里处理的就是磁盘上文件

文件名:文件名包含3部分——文件路径+文件名主干+文件后缀,例如:c:\code\test.txt

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

根据数据的存储格式,数据文件分为文本文件二进制文件两类

数据在内存中以二进制的形式存储,若不加转换的输出到外存的文件中,就是二进制文件;
如果在存储前转换为ASCII码,就是文本文件

那一个数据在文件中是如何存储的呢?
字母一律以ASCII形式储存,数值型数据既可以用ASCII形式存储,也可以用二进制形式存储,例如:
整数10000,如果以ASCII码形式输出到磁盘,则磁盘中占用5个字节(每个字符一个字节),而二进制形式输出,则占4个字节
在这里插入图片描述
代码如下:

#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;
}

打开方式如下:
在这里插入图片描述
在这里插入图片描述

4. 对文件的了解

4.1 流和标准流

流:
程序的数据有时需要输出到各种外部设备,也需要从外部设备获取数据,不同的外部设备的输入输出操作各不相同,为了方便我们对各种设备进行方便的操作,我们抽象除了流的概念。我们可以把流看成是流淌着字符的河。

C程序针对文件、画面、键盘等的数据输入输出操作都是通过流操作的;要想向流里写写数据,或者从流中读取数据,要先打开流,再操作

标准流:
C语言程序在启动的时候,默认打开了三个流:

  • stdin:标准输入流,在大多数的环境中从键盘输入,scanf函数就是从标准输入流中读取数据的
  • stdout:标准输出流,大多数的环境中输出至显示器界面,printf函数就是将信息输出到标准输出流中
  • stderr:标准错误流,大多数环境中输出到显示器界面

这三个流的类型是FILE*,通常被称为文件指针

4.2 文件指针

缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”

每个被使用的文件都会在内存中开辟一个文件信息区,存放文件的相关信息。信息被保存在一个结构体变量中,该结构体类型由系统统一申明,取名FILE(不同的C编译器的FILE类型包含的内容不一定相同,但大同小异)
比如,VS2013编译环境提供的stdio.h头文件中的申明:

struct _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;

我们可以通过文件指针变量间接找到与它关联的文件:
在这里插入图片描述

4.3 文件的打开和关闭

文件在读写前要先打开,使用后要关闭

ANSI C规定使用fopen函数打开文件,fclose关闭文件,函数形式如下:

FILE* fopen(const char* filename,const char * mode);// 返回指向该文件的文件指针
int fclose(FILE * stream);

mode表示文件的打开模式,如下:

文件使用方式含义如果指定文件不存在
“r” (只读)为了输入数据,打开一个已经存在的文本文件出错
“w” (只写)为了输出数据,打开一个文本文件建立一个新文件
“a” (追加)向文本文件末尾添加数据建立一个新文件
“rb” (只读)为了输入数据,打开一个二进制文件出错
“wb” (只写)为了输出数据,打开一个二进制文件建立一个新文件
“ab” (追加)向一个二进制文件末尾添加数据建立一个新文件
“r+” (读写)为了读和写,打开一个文本文件出错
“w+” (读写)为了读和写,建立一个新文件建立一个新文件
“a+” (读写)打开一个文件,在文件尾进行读写建立一个新文件
“rb+” (读写)为了读和写,打开一个二进制文件出错
“wb+” (读写)为了读和写,建立一个新的二进制文件建立一个新文件
“ab+” (读写)打开一个二进制文件,在文件尾进行读和写建立一个新文件
示例代码:
#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 顺序读写函数

函数名功能适⽤于
fgetc字符输⼊函数所有输⼊流
fputc字符输出函数所有输出流
fgets⽂本行输⼊函数所有输⼊流
fputs⽂本行输出函数所有输出流
fscanf格式化输⼊函数所有输⼊流
fprintf格式化输出函数所有输出流
fread⼆进制输⼊⽂件输⼊流
fwrite⼆进制输出⽂件输出流

所有输入流指标准输入流和其他输入流(如文件输入流);所有输出流指标准输出流和其他输出流(如文件输出流)

5.2 对比一组函数

函数功能
scanf从标准输入流上读取格式化的数据
fscanf从指定输入流上读取格式化的数据
sscanf在字符串中读取格式化的数据
函数功能
printf把数据以格式化的形式打印在标准输入流上
fprintf把数据以指定的形式打印在指定的输出流上
sprintf把格式化的数据转化成字符串

6. 文件的随机读写

6.1 fseek

根据文件指针的位置和偏移量来定位文件指针(文件内容的光标)

int fseek(FILE * stream,long int offset,int origin);

示例:

#include<stdio.h>int main()
{FILE * pFile;pFile=fopen("example.txt","wb");fputs("This is an apple.",pFile);fseek(pFile,9,SEEK_SET);fputs(" sam",pFile);fclose(pFile);return 0;
}

6.2 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-portablesize=ftell (pFile);fclose (pFile);printf ("Size of myfile.txt: %ld bytes.\n",size);}return 0;
}

6.3 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;
}

7. 文件读取结束的判定

7.1 被错误使用的feof

foef 的作用是:当文件读取结束的时候,判断读取结束的原因是否是遇到文件尾结束

所以在文件读取过程中,不能用 foef 函数的返回值来判断文件是否结束

7.2 正确判定方式

1、文本文件判定方式:判断返回值是否为EOF(fgetc),或者NULL(fgets)。如:

  • fgetc判断是否为EOF
  • fgets判断返回值是否为NULL

2、二进制文件判定方式:判断返回值是否小于实际要读的个数。如:

  • fread判断返回值是否小于实际要读的个数

文本文件实例:

#include <stdio.h>
#include <stdlib.h>
int main(void)
{int c; // 注意:int,⾮char,要求处理EOFFILE* fp = fopen("test.txt", "r");if(!fp) {perror("File opening failed");return EXIT_FAILURE;}//fgetc 当读取失败的时候或者遇到⽂件结束的时候,都会返回EOFwhile ((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 handlingif (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编译系统决定的

在这里插入图片描述

#include <stdio.h>
#include <windows.h>
//VS2022 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;
}

因为有缓冲区的存在,C语言在操作文件的时候,需要刷新缓冲区或在文件操作结束时关闭文件。如果不,可能会导致读写文件的问题

http://www.xdnf.cn/news/672373.html

相关文章:

  • 【深度学习-Day 18】从SGD到Adam:深度学习优化器进阶指南与实战选择
  • 【JavaSE】枚举和注解学习笔记
  • 高考加油(Python+HTML)
  • 青少年编程与数学 02-020 C#程序设计基础 07课题、控制结构
  • Android设置界面层级为最上层实现
  • 【数据集】EarthExplore下载Landsat LST 数据
  • Java网络编程性能优化
  • 【Vue Vapor Mode :技术突破与性能优化的可能性】
  • 嵌入式学习--江协stm32day1
  • 第9章:网络协议栈与 IoT 通信
  • React 第四十七节 Router 中useLinkClickHandler使用详解及开发注意事项案例
  • React 第四十八节 Router中 useMatch 的使用详细介绍及案例分析
  • React---day2
  • 微服务及容器化设计--可扩展的架构设计
  • Python 中的 for 循环:从基础到高级应用的全面解析
  • WPF【09】WPF基础入门 (三层架构与MVC架构)
  • 沈阳城市空气质量综合评价系统/答辩以及详细讲解
  • 基于cornerstone3D的dicom影像浏览器 第二十四章 显示方位、坐标系、vr轮廓线
  • Python requests
  • App Runner和Task Pipeline中的数据库连接管理指南
  • 【数据结构】树形结构--二叉树
  • U-Boot ARMv8 平台异常处理机制解析
  • Android studio 查看aar源码出现/* compiled code */
  • 基于 MindQuantum 记录线路作用下基底态的变化过程
  • 让jupyter notebook显示目录
  • 大模型应用:开发移动端页面个人中心页面提示词
  • 基于大模型预测视神经脊髓炎的技术方案大纲
  • Ubuntu 20.04 下 OpenCV 4.5.4 编译安装与系统默认 OpenCV 4.2 共存切换指南【2025最新版!!!】
  • Elasticsearch创建快照仓库报错处理
  • 嵌入式学习--江协stm32day3