适合零基础小白,循序渐进学习--文件操作--相关函数

news2024/10/1 21:34:06

目录

一. 前言

二. 正文

2. 1什么是文件

(1)程序文件

(2)数据文件

2.2文件名

2.3文件类型

2.4 文件缓冲区

2.5 文件指针

 2.7 文件操作函数

(1) fopen()--- 打开文件

(2) fclose()--- 关闭文件

2.7 文件的顺序读写,各函数运用

(1)fgetc与fputc --------  字符输入输出函数

   (2)fgets与fputs--- 文本输入输出函数

 (3) sscanf 与 sprintf--- 格式输入输出函数

 (4)fread 与 fwrite--- 二进制输入输出函数

2.8文件结束判定

三. 结语


一. 前言

二. 正文

2. 1什么是文件

磁盘上的文件是文件。 但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件

(1)程序文件

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

(2)数据文件

   文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。
本节讨论的是数据文件。 
   在以前所处理数据的输入输出都是以终端为对象的,即从终端的键盘输入数据,运行结果显示到显示器上。 其实有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使用,这里处理的就是磁盘上文件。

2.2文件名

一个文件要有一个唯一的文件标识,以便用户识别和引用。
文件名包含 3 部分:文件路径 + 文件名主干 + 文件后缀
例如: c:\code\test.txt
为了方便起见,文件标识常被称为 文件名

2.3文件类型

根据数据的组织形式,数据文件被称为 文本文件 或者 二进制文件
  • 数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件
  • ASCII字符的形式存储的文件就是文本文件。(如果要求在外存上以ASCII码的形式存储,则需要在存储前转换)

2.4 文件缓冲区

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

2.5 文件指针

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

 2.7 文件操作函数

(1) fopen()--- 打开文件

FILE * fopen ( const char * filename , const char * mode );
filename:  路径 字符串,如果与源文件在同一个文件夹,可以直接填  文件名+ 文件类型。
mode:      打开方式,指定字符。
这里对常见的打开方式列举一下:
文件使用方式                        含义                                                    如果文件不存在
“r” (只读)        为了输入数据,打开一个已经存在的文本文件       出错
“w” (只写)       为了输出数据,打开一个文本文件                         建立一个新的文件
“a” (追加)        向文本文件尾添加数据                                          出错
“rb” (只读)       为了输入数据,打开一个二进制文件                    出错
“wb” (只写)     为了输出数据,打开一个二进制文件                     建立一个新的文件

(2) fclose()--- 关闭文件

int fclose ( FILE * stream );

运用:
/* fopen fclose example */
#include <stdio.h>
int main ()
{
  FILE * pFile = fopen ("myfile.txt","w"); // 如果没有此文件,就会创建此文件。 
  //                                          同时,会销毁原文件数据
  if (pFile ==NULL)
 {
    perror("fopen");  // 打印fopen失败原因
    exit(-1);
 }
  .....录入数据
  fclose(pFile);
  pFile = null; // 防止后边解应用
  return 0;
}

2.7 文件的顺序读写,各函数运用

功能                                             函数名                                            适用于
字符输入函数                             fgetc                                               所有输入流
字符输出函数                             fputc                                               所有输出流
文本行输入函数                         fgets                                               所有输入流
文本行输出函数                         fputs                                               所有输出流
格式化输入函数                         fscanf                                             所有输入流
格式化输出函数                         fprintf                                             所有输出流
二进制输入                                fread                                              文件
二进制输出                                fwrite                                              文件

(1)fgetc与fputc --------  字符输入输出函数

代码运用举例:
#include<stdio.h>
int main()
{
	FILE* a =  fopen("Test.txt", "w");
	if (a ==NULL)
	{
		perror("fopen");
		exit(-1);
	}
	fputc('a', a);  // 输出
	fputc('b', a);
	fputc('c', a);
	fputc('d', a);
	fputc('e', a);
	fclose(a);

	a = fopen("Test.txt", "r");  // 重新打开文件
	int ch = 0;
	while ((ch = fgetc(a)) != EOF)
	{
		printf("%c ", ch);
	}
	fclose(a);  // 关闭文件
	a = NULL;
	return 0;
}

结果:


(2)fgets与fputs--- 文本输入输出函数

代码举例:

#include<stdio.h>

int main()
{
	FILE* a = fopen("test.txt", "w");// 以 只写 方式打开文件
	if (a ==NULL)
	{
		perror("fopen");
		exit(-1);
	}
	fputs("abcdefgh", a);
	fclose(a);

	a = fopen("test.txt", "a");  // 重新以  增添  方式打开文件
	fputs("ahuang", a);
	fclose(a);

	a = fopen("test.txt", "r");  // 重新以 只读 方式打开文件
	char* c[20] = { 0 };
	fgets(c, 20, a); // 会少打印一个字符,因为要留个位置给\0。
	printf("%s", c);
	fclose(a);
	a = NULL;
	return 0;
}

结果:


 (3) sscanf 与 sprintf--- 格式输入输出函数

#include<stdio.h>
struct MyStruct
{
	char name[20];
	int age;
	double size;
};

int main()
{
	struct MyStruct s = { "ahuang", 20 ,60};
	// 打开文件
	FILE* a = fopen("Test2.txt", "w");
	if (a == NULL)
	{
		perror("fopen");
		exit(-1);
	}
	fprintf(a, "%s %d %lf", s.name, s.age, s.size);
	fclose(a);

	a = fopen("Test2.txt", "r");
	struct MyStruct c = { 0 }; // 接收文件读取的数据
	fscanf(a,"%s %d %lf", c.name, &(c.age), &(c.size));
	fprintf(stdout, "%s %d %lf\n", c.name, c.age, c.size); // 打印到屏幕
    // 上一段等价于 printf("%s %d %lf", c.name, c.age, c.size);
	fclose(a);
	a = NULL;
	return 0;
}

结果:

这里补充一下 ,当一个C程序启动时,以下三个信息流是默认打开的:

标准输入信息流,  stdin,   来自键盘

标准输出信息流,  stdout,  输出给显示器

标准错误信息流,     stderr ,  输出给显示器

 (4)fread 与 fwrite--- 二进制输入输出函数

代码运用如下:

#include<stdio.h>
struct MyStruct
{
	char name[20];
	int age;
	double size;
};

int main()
{
	FILE* p1 = fopen("Text.txt", "w");
	struct MyStruct k = { "阿黄", 20, 89 };
	fwrite(&k, sizeof(struct MyStruct), 1, p1);
	fclose(p1);

	p1 = fopen("Text.txt", "r");
	struct MyStruct z = { 0 };
	fread(&z, sizeof(struct MyStruct), 1, p1);
	printf("%s %d %lf", z.name, z.age, z.size);
	return 0;
}

结果:

 这里补充一下:为什么二进制输入输出数据在记事本里会乱码原因:

     展现形式不匹配

     解析:前三种输入输出方式是用ASCII码形式保存,是文本文件;最后一种是用二进制形式保存,而以记事本打开数据本身要是ASCII码形式保存,所以二进制形式会乱码。

2.8文件结束判定

被错误使用的 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))     // 如果ferror返回非0,那么中间出现错误。
       puts("I/O error when reading");
   else if (feof(fp))  // 判断是否遇到EOF才结束,是就是正常结束。
       puts("End of file reached successfully");
   fclose(fp);
}

三. 结语

    本小节就到这里了,感谢小伙伴的浏览,如果有什么建议,欢迎在评论区评论,如果给小伙伴带来一些收获请留下你的小赞,你的点赞和关注将会成为博主创作的动力。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/437519.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ArcGIS制图技巧

目录 1、经纬度网置于底层 2、设置经纬网刻度为英文 3、设置左右经纬度为垂直&#xff0c;不显示分秒&#xff0c;以及改变字体等 4、拖动制图中的图层 5、设置经纬度网为曲线 6、根据经纬度导入样本点 1、经纬度网置于底层 最初我们的网格是下图这个样子&#xff1a; 然…

智慧化工业企业能耗管理系统平台的构建

节能监测技术是节能减排、有效节能的重要支撑技术&#xff0c;当前能源资源日渐枯竭的紧张形势下&#xff0c;节能意识的增强以及各行业对节能技术的需求不断增加&#xff0c;使得节能技术的开发越来越受到人们的重视。工业企业能耗监测系统的构建是按照“统一规划、分期分批”…

头像展示样式

先上效果图 再上代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title><style>body {align-items: center;background-color: #f2f2f2;display: flex;font-fa…

代码随想录算法训练营第三十天| 332.重新安排行程、回溯总结

文章目录 332.重新安排行程:star:回溯总结:star:1.组合问题2.切割问题3.子集问题4.排列问题5.棋盘问题(未完待续)6.复杂度分析 332.重新安排行程⭐️ 链接:代码随想录 本题是一道困难题&#xff0c;其实困难点也就在容器的选择和使用上 结果集采用数组存取即可操作集要用一个m…

算法篇——栈与队列大集合(js版)

232.用栈实现队列 请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作&#xff08;push、pop、peek、empty&#xff09;&#xff1a; 实现 MyQueue 类&#xff1a; void push(int x) 将元素 x 推到队列的末尾 int pop() 从队列的开头移除并返回元素 int…

SpringBoot集成Redis及问题解决

SpringBoot集成Redis 此篇文章为SpringBoot集成Redis的简单介绍&#xff0c;依赖、序列化操作、工具类都可以在后面的实操中直接搬运使用或者在此基础上进行改进使用 1、集成Redis 1.1、新建SpringBoot项目 新建项目这边就不一一介绍了&#xff0c;大家如果还有不会的可以自行…

【LeetCode】144.二叉树的前序遍历

1.问题 给你二叉树的根节点 root &#xff0c;返回它节点值的 前序 遍历。示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,2,3] 示例 2&#xff1a; 输入&#xff1a;root [] 输出&#xff1a;[] 示例 3&#xff1a; 输入&#xff1a;root [1]…

七大排序的时间复杂度和空间复杂度

文章目录 七大排序的分类时间复杂度空间复杂度稳定性直接插入排序希尔排序选择排序堆排序冒泡排序快速排序归并排序总结 七大排序的分类 时间复杂度 时间复杂度是指一个程序中基本语句被执行的次数,一般认为是最坏情况。 空间复杂度 空间复杂度是指在一个程序执行时要额外开…

美团DAT:A Dual Augmented Two-tower Model for Online Large-scale Recommendation

A Dual Augmented Two-tower Model for Online Large-scale Recommendation 美团的对偶增强双塔为了user塔和item塔操碎了心&#xff0c;众所周知&#xff0c;双塔的一个大毛病就是item和user的交叉太晚&#xff0c;重要的信息经过层层神经网络的抽象提取&#xff0c;有些重要…

15、异常处理

文章目录 1、错误处理1、默认规则2、定制错误处理逻辑3、异常处理自动配置原理4、异常处理步骤流程 【尚硅谷】SpringBoot2零基础入门教程-讲师&#xff1a;雷丰阳 笔记 路还在继续&#xff0c;梦还在期许 1、错误处理 1、默认规则 默认情况下&#xff0c;Spring Boot提供/er…

傻傻的分不清 Comparator 和 Comparable 接口? 两分钟弄懂~

目录 一、Comparable 接口 二、Comparator 接口 一、Comparable 接口 Comparable 是一个排序接口&#xff08;意味着该支持排序&#xff09;&#xff0c;可以看作内比较器&#xff0c;也就是说可以和自己比较&#xff0c;通常用来自己属性与自己属性进行比较&#xff0c;最后通…

百年孤独 -- 有感

《创世记》之后&#xff0c;首部值得全人类阅读的文学巨著。 — 纽约时报 加西亚 马尔克斯以小说作品创建了一个自己的世界&#xff0c;一个浓缩的宇宙&#xff0c;其中喧嚣纷乱却又生动可信的现实&#xff0c;映射了一篇大陆及其人民的富足与贫困。 — 诺贝尔文学奖颁奖辞 马…

wordpress+apache搭建问题总结

访问首页出现服务器目录 需要到apache的httpd.conf 添加index.php默认值 更新失败。 此响应不是合法的JSON响应 大多出现于修改固定连接后 打开httpd.conf文件,把AllowOverride None修改为AllowOverride All,重启apache即可解决 AllowOverride Noneyi意味着忽略.htaccess文件…

VS2022中创建C++SDK库(dll动态库)并调用SDK库(dll动态库)

VS2022中创建CSDK库(dll动态库)并调用SDK库(dll动态库) 一、说明 通过前两篇的文章我们知道了封装一个Qt下的SDK库&#xff08;dll动态链接库&#xff09;和封装Pimpl模式。 Qt创建SDK库(dll动态库)并调用SDK库(dll动态库) SDK(动态链接库dll)的封装技巧 本篇介绍在C VS下…

深度解析动态分配内存管理

目录 ​编辑 一. 前言 二.正文 2.0 计算机中的内存 2.1 动态申请函数头文件 2.2 malloc函数 2.3 free函数 2.3 calloc函数 2.4 realloc函数 2.5 经典笔试题 1. 2. 2.6 柔性数组 三.结语 一. 前言 本小节跟大家分享动态内存管理的知识&#xff0c;希望能给大家带…

2020年全国10m分辨率地表覆盖数据集

1.数据概况 地球观测技术的进步使得土地覆盖制图的空间分辨率不断提高&#xff0c;但这也导致了高分辨率影像样本收集和更新变得更加耗时。本研究提出了一种基于样本噪声学习的跨分辨率土地覆盖制图框架&#xff0c;利用现有低分辨率产品生成大量高分辨率影像标签用于深度语义…

【Python从入门到进阶】17、文件的读写操作

接上篇《16、文件的打开和关闭》 上一篇我们学习了Python打开和关闭文件的操作。本篇我们主要学习Python文件对象包含哪些方法&#xff0c;并学习文件如何进行读、写、定位和删除。 一、文件对象的属性和方法 通过上一篇的学习我们知道&#xff0c;open函数会返回给我们一个文…

一个简单的盒模型嵌套---一个大盒子里面嵌套两个中盒子,右边中盒子嵌套六个小盒子

html代码&#xff1a; <div class"box1"><div class"box2"></div><div class"box3"><div class"box4"></div><div class"box5"></div><div class"box6">&…

apple pencil值不值得购买?ipad平替电容笔安利

自从Apple Pencil上市后&#xff0c;平替电容笔在国内的销量&#xff0c;也是随之而增加&#xff0c;虽然Apple Pencil的性能不错&#xff0c;但由于其的价格太高&#xff0c;所以不少普通用户都买不起。现在市面上有很多能够代替苹果原装笔的平替电容笔。在这里&#xff0c;我…

Debezium系列之:自定义表级别Topic的名称

Debezium系列之:自定义表级别Topic的名称 一、需求背景二、去掉部分topic名称的方法三、debezium具体实现方法四、完整配置一、需求背景 debezium表级别Topic命名规则为:topic.prefix参数的值.数据库名称.表的名称现在支持实时需求,命名名称需要修改为topic.prefix参数的值.…