C语言—文件操作(学好文件操作,再也不用担心数据丢失)

news2025/1/10 12:06:14

专栏:C语言
个人主页:HaiFan.
专栏简介:本专栏主要更新一些C语言的基础知识,也会实现一些小游戏和通讯录,学时管理系统之类的,有兴趣的朋友可以关注一下。

文件操作

  • 前言
  • 一、为什么使用文件
  • 二、什么是文件
    • 1.程序文件
    • 2.数据文件
    • 3.文件名
  • 三、文件的打开和关闭
    • 1.文件指针
    • 2.文件的开关
    • 3.文件的权限
    • 4.文件的顺序读写
  • 四、文件的随机读写
    • 1.fseek
    • 2.ftell
    • 3.rewind
  • 五、文本文件和二进制文件
  • 六、文件读取结束的判定
    • 1.被错误使用的feof
  • 七、文件缓冲区

前言


在这里插入图片描述


一、为什么使用文件

前面学习结构体的时候,写了一个通讯录的程序,但是当通讯录运行的时候,把数据写再来程序里,但此时的数据是放在内存中的,当程序结束的时候,内存中的数据被销毁了,通讯录中的数据也就不存在了,再次运行通讯录的时候,又要重新写入数据,这样的通讯录是不完整的通讯录。
想把数据保存下来就要想其他办法,这就涉及到了数据持久化的问题,我们一般数据持久化的方法有,把数据存放在磁盘文件、存放到数据库等方式。
使用文件操作,我们可以将数据直接存放到电脑上的硬盘上,做到了数据的持久化。

二、什么是文件

磁盘上的文件是文件。
但是在程序设计中,我们一般谈的文件有两种,程序文件,数据文件(从文件的功能的角度来划分的)

1.程序文件

包括源程序文件(后缀名.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境下后缀为.exe)
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

2.数据文件

文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。

3.文件名

一个文件要有一个唯一的文件标识,以便用户识别和引用,文件名包含三部分:文件路径+文件名主干+文件名后缀
如:

c:\code\test.txt

为了方便起见,文件标识常被称为文件名。

三、文件的打开和关闭

1.文件指针

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

在这里插入代码片

不同的C语言编译器的FILE类型包含的内容不完全相同,但是大同小异。
每当打开一个文件的时候,系统会根据文件的情况自动下黄健一个FILE结构的变量,并填充其中的信息。
一般都是通过一个FILE的指针来维护这个FILE结构的变量。

FILE* pf;//文件指针变量,创建一个FILE的指针变量

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

2.文件的开关

文件在读写前,要先打开文件,结束使用的时候关闭文件。
在编写程序的时候,在打开文件的同时,都会返回一个FILE*的指针变量指向该文件,相当于建立了指针和文件的关系。

//打开文件
FILE * fopen ( const char * filename, const char * mode );
//关闭文件
int fclose ( FILE * stream );


在这里先创建了一个文本文档:test.txt

#include <stdio.h>

int main()
{
	FILE* fp = fopen("test.txt", "w");//文件的打开
	if (fp == NULL)
	{
		perror("fopen");
		return;
	}
	fclose(fp);//文件的关闭
	return 0;
}

3.文件的权限

文件的使用方式含义若文件不存在
r只读—为了输入数据,打开一个已经存在的文本文件出错
w只写—为了输出数据,打开一个文本文件建立一个新的文件
a追加—向文本文件尾添加数据建立一个新的文件
rb只读—为了输入数据,打开一个二进制文件出错
wb只写—为了输出数据,打开一个二进制文件建立一个新的文件
ab追加—向一个二进制文件尾添加数据出错
r+读写—为了读和写,打开一个文本文件出错
w+读写—为了读和写,创建一个文本文件建立一个新文件
a+读写—打开一个文件,在文件尾进行读写建立一个新文件
rb+读写—为了读和写打开一个二进制文件出错
wb+读写—为了读和写,新建一个新的二进制文件建立一个新的文件
ab+打开一个二进制文件,在文件尾进行读和写建立一个新的文件

4.文件的顺序读写

功能函数名适用于
字符输入函数fgetc所有输入流
字符输出函数fputc所有输出流
文本行输入函数fgets所有输入流
文本行输出函数fputs所有输出流
格式化输入函数fscanf所有输入流
格式化输出函数fprintf所有输出流
二进制输入fread文件
二进制输出fwrite文件
int fgetc ( FILE * stream );
int fputc ( int character, FILE * stream );

fgetc就是从流中获取字符,并自动指向以获取字符的下一个位置。若流位于文件末尾,则返回EOF。
fputc就是将字符写入流内,也就是写入文件内。

#include <stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char aa;
	for (aa = 'a'; aa <= 'z';aa++)
	{
		fputc('a', pf);
	}
	fclose(pf);
	pf = NULL;
	return 0;
}

在这里插入图片描述

char * fgets ( char * str, int num, FILE * stream );
int fputs ( const char * str, FILE * stream );

fgets就是从流中获取字符,并将其作为C字符串存到str中,直到读取num个字符或到达换行符或文件末尾。到家文件末尾返回EOF
fputs就是将str指向流的字符串写入流。意思就是把str存到文件里。

#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("hao hao xue xi tian tian xiang shang", pf);
	//printf("%c ",fgetc(pf));

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

在这里插入图片描述
利用fputs读取字符串,遇到\0停止。

#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//fputs("hao hao xue xi tian tian xiang shang", pf);
	char str[100];
	printf("%s ",fgets(str,100,pf));

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

在这里插入图片描述

int fscanf ( FILE * stream, const char * format, ... );
int fprintf ( FILE * stream, const char * format, ... );

fsanf从流中读取数据,并根据参数的格式将其存储到文件里。
fprintf按照格式,将内容存到文件里。

#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fprintf(pf, "%s %d", "好好学习,找个好的offer",100);

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

在这里插入图片描述

#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char str[100];
	int t = 0;
	fscanf(pf, "%s %d", str, &t);
	printf("%s\n%d", str, t);
	fclose(pf);
	pf = NULL;
	return 0;
}

在这里插入图片描述

size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );
size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );

fwrite

  1. 第一个参数:要写入文件的东西的地址
  2. 第二个参数:要写入的每个元素的大小
  3. 第三个参数:元素个数
  4. 第四个参数:流,就是把元素写进文件里

fread

  1. 第一个参数:指向大小至少为 (size*count) 字节的内存块的指针。
  2. 第二个参数:要读取的每个元素的大小
  3. 第三个参数:元素个数
  4. 第四个参数:指向指定输入流的 FILE 对象的指针。
#include <stdio.h>

struct stu
{
	char name[10];
	int age;
	int score;
};

int main()
{
	struct stu a = { "牛马", 18, 0 };
	FILE* pf = fopen("test.txt", "rb+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	
	fwrite(&a, sizeof(a), 1, pf);
	fclose(pf);
	pf = NULL;
	return 0;
}

在这里插入图片描述
因为这个函数是二进制输出函数,所以后面的看不懂正常,编译器能看得懂就行。

#include <stdio.h>
struct stu
{
	char name[10];
	int age;
	int score;
};
int main()
{
	struct stu a = { "牛马", 18, 0 };
	struct stu b;

	FILE* pf = fopen("test.txt", "rb+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//fwrite(&a, sizeof(a), 1, pf);
	fread(&b, sizeof(b), 1, pf);
	printf("%s %d %d", b.name, b.age, b.score);
	fclose(pf);
	pf = NULL;
	return 0;
}

在这里插入图片描述

四、文件的随机读写

1.fseek

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

int fseek ( FILE * stream, long int offset, int origin );
#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char c = 'a';
	for (c = 'a'; c <= 'z'; c++)
	{
		fputc(c, pf);
	}
	//fseek(pf, -24, SEEK_CUR);
	char ed = fgetc(pf);
	char cur = fgetc(pf);
	char st = fgetc(pf);

	//fseek()
	printf("%c %c %c", ed, cur, st);
	fclose(pf);
	pf = NULL;
	return 0;
}

这个代码其实是并没有任何输出的,因为在进行fputc函数读入的时候,始终都是在文件尾部追加的字符,所以后面在进行fgetc的时候,不会打印出来东西。
这个时候就用到了fseek函数。
在这里插入图片描述

#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char c = 'a';
	for (c = 'a'; c <= 'z'; c++)
	{
		fputc(c, pf);
	}
	fseek(pf, -24, SEEK_CUR);
	char ed = fgetc(pf);
	char cur = fgetc(pf);
	char st = fgetc(pf);

	//fseek()
	printf("%c %c %c", ed, cur, st);
	fclose(pf);
	pf = NULL;
	return 0;
}

在这里插入图片描述
这个SEEK_SET就是在文件开头的位置偏移,如果代码中fseek(pf, 0, SEEK_SET);是这样写的,就会打印a,b,c。如果是fseek(pf, -10, SEEK_CUR);这样写,就相当于在文件的当前位置,向左偏移10个位置,打印的就是qrs。
最后一个同理,负数向左偏移,正数向后偏移。

2.ftell

返回文件指针相对于其实位置的偏移量

long int ftell ( FILE * stream );
#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char c = 'a';
	for (c = 'a'; c <= 'z'; c++)
	{ 
		fputc(c, pf);
	}
	int size = ftell(pf);
	printf("%d", size);
	fclose(pf);
	pf = NULL;
	return 0;
}

在这里插入图片描述

3.rewind

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

void rewind ( FILE * stream );
#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char c = 'a';
	for (c = 'a'; c <= 'z'; c++)
	{
		fputc(c, pf);
	}
	
	int size = ftell(pf);
	printf("%d\n", size);
	rewind(pf);
	size = ftell(pf);
	printf("%d", size);
	fclose(pf);
	pf = NULL;
	return 0;
}

在这里插入图片描述

五、文本文件和二进制文件

根据数据的组织形式,数据文件被称为文本文件或者二进制文件。数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件。
一个数据在内存中是怎么存储的呢?
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。如有整10000,如果以ASCII码的形式输出到磁盘,则磁盘中占用5个字节(每个字符一个字节),而二进制形式输出,则在磁盘上只占4个字节.

六、文件读取结束的判定

1.被错误使用的feof

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

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

int main()
{
	FILE* pf = fopen("test.txt", "r+");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	char c;
	while ((c = fgetc(pf)) != EOF)
	{
		putchar(c);
	}
	puts("");
	if (feof(pf))
	{
		puts("I/O error when reading");
	}
	else if (ferror(pf))
	{
		puts("End of file reached successfully");
	}
	fclose(pf);
	pf = NULL;
	return 0;
}

七、文件缓冲区

ANSIC 标准采用“缓冲文件系统”处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区或者在文件操作结束的时候关闭文件。如果不做,可能导致读写文件的问题。

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

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

相关文章

Leetcode:538. 把二叉搜索树转换为累加树(C++)

目录 问题描述&#xff1a; 实现代码与解析&#xff1a; 反中序遍历法&#xff1a; 原理思路&#xff1a; 迭代&#xff1a; 原理思路&#xff1a; 问题描述&#xff1a; 给出二叉 搜索 树的根节点&#xff0c;该树的节点值各不相同&#xff0c;请你将其转换为累加树&…

共享模型之管程(九)

1.ReentrantLock可重入锁 1.1.简介 1>.可重入是指同一个线程如果首次获得了这把锁,那么由于它是这把锁的拥有者,因此该线程有权利(/优先)再次获取这把锁;如果是不可重入锁,那么第二次获得锁时,自己也会被锁挡住; ReentrantLock底层也是基于Monitor对象实现的,只不过它是在…

java支持的数据类型2023006

Java语言支持的类型分为两类&#xff1a;基本类型&#xff08;Primitive Type&#xff09;和引用类型&#xff08;Reference Type&#xff09;。 –基本类型包括boolean类型和数值类型。数值类型有整数类型和浮点类型。整数类型包括byte、short、int、long、char&#xff0c;浮…

dp(九)不同的子序列

不同的子序列_牛客题霸_牛客网【牛客题霸】收集各企业高频校招笔面试题目&#xff0c;配有官方题解&#xff0c;在线进行百度阿里腾讯网易等互联网名企笔试面试模拟考试练习,和牛人一起讨论经典试题,全面提升你的技术能力https://www.nowcoder.com/practice/ed2923e49d3d495f83…

Vue知识系列-axios

一、axios基础知识 axios是独立于vue的一个项目&#xff0c;基于promise用于浏览器和node.js的http客户端。 在浏览器中可以帮助我们完成ajax请求的发送在node.js中可以向远程接口发送请求 二、axios应用场景 三、axios的使用 第一步&#xff1a;创建html文件&#xff0c;在…

6.见过最强的后台管理系统

芋道&#xff0c;以开发者为中心&#xff0c;打造中国第一流的快速开发平台&#xff0c;全部开源&#xff0c;个人与企业可 100% 免费使用。 架构图 管理后台的 Vue3 版本采用 vue-element-plus-admin &#xff0c;Vue2 版本采用 vue-element-admin 管理后台的移动端采用 uni-…

虹科方案 | 医药行业专用ECOLOG即插即用室内和设施环境监控系统

虹科ELPRO ECOLOG Plug&Play 连续监测解决方案采用功能强大的新ECOLOG-PRO硬件模块和ECOLOG-PRO Base服务器解决方案。即插即用是虹科ELPRO室内和设施环境监测解决方案增强型系列的一部分。ECOLOG即插即用是一种多功能&#xff0c;经济的系统&#xff0c;专为需要监测少于5…

转换流:代码与文件编码不一致读取乱码的问题

字符输入转换流&#xff1a;package com.gao;import java.io.*;public class transfer_stream {public static void main(String[] args) throws Exception {InputStream in new FileInputStream("E:\\黑马教育\\demo.txt");Reader re new InputStreamReader(in,&q…

Postman(07)Postman+OMySQL操作数据库

一、本文解决的问题 ● Postman断言需要查询数据存储情况&#xff1b; ● 部分业务需要从数据库中获取验证码&#xff08;例如登录&#xff09;&#xff1b; ● 自动化场景需要执行数据正确性比对。 二、OMySQL连接数据库 1、OMySQL安装与启动 1.1 安装node.js 这个百度一…

【windows】VMware Workstation 不可恢复错误:(vmui) 错误代码0xc0000094

【问题描述】 打开需要启动的虚拟机后&#xff0c; 修改设备参数的时候&#xff0c;提示&#xff1a; VMware Workstation 不可恢复错误: (wmui) Exception 0xc0000094 has occurred。 【解决方法】 由于搭建的虚拟机环境一直在使用&#xff0c; 后来发现是VMware升级到17.0后…

【回眸】牛客网刷刷刷(三) Linux(续)软件工程(专题)

前言 近期的牛客网学习内容有Linux专题&#xff0c;以下记录一些笔试题中的Linux的知识点。 知识点 -u 用户号 指定用户的用户号&#xff1b;因为系统用户的用户号为 0&#xff0c;故指定用户号为 0&#xff1b; 如果同时有 -o 选项&#xff0c;则可以重复使用其他用户的标识…

马来酰亚胺-二硫键-活性酯;MAL-SS-NHS结构式

英 文 &#xff1a;MAL-SS-NHS 中文&#xff1a;马来酰亚胺-双硫键-活性酯 存储条件&#xff1a;-20C&#xff0c;避光&#xff0c;避湿 用 途&#xff1a;仅供科研实验使用&#xff0c;不用于诊治 外观: 固体或粘性液体&#xff0c;取决于分子量 注意事项&#xff1a;取用…

【机器学习】逻辑回归(理论)

逻辑回归&#xff08;理论&#xff09; 目录一、概论1、何为逻辑回归2、映射函数的引入3、伯努利分布二、损失函数的推导三、用逻辑回归实现多分类1、间接法&#xff1a;HardMax2、直接法&#xff1a;SoftMaxⅠ SoftMax的引入Ⅱ SoftMax的计算Ⅲ 引入SoftMax后的损失函数&#…

声网许振明:RTC 场景 UHD 视频应用和探索

大家好&#xff0c;我是声网的视频工程师许振明&#xff0c;今天跟大家主要介绍一下声网在 RTC 场景 UHD 视频的应用和探索。主要基于声网 HFR 和 VDR 两个系统来展开分享。 随着 RTC 技术的发展和应用&#xff0c;越来越多的场景都需要接入 RTC 的能力。尤其是随着编码技术、…

【Linux】Ubuntu20.04.5安装ROS1【教程】

文章目录【Linux】Ubuntu20.04.5安装ROS1【教程】写在前面&#xff0c;一些特别注意的点安装步骤大纲1.选择软件镜像源&#xff08;可以省略&#xff0c;如果不成功再试这一步&#xff09;2.添加ROS软件源和密钥3.安装ROS14.设置环境变量5.安装rosinstall6.检查安装是否成功Ref…

基于uds和modbus协议的bootloader对比

2022年参与了两个bootloader项目的开发&#xff0c;两个项目使用的芯片、通讯驱动不同&#xff0c;因此传输协议也不同&#xff0c;年底了将两个项目做一个对比&#xff0c;也顺便加深下对bootloader的理解。相同点&#xff1a;modbusuds刷新流程刷新流程大同小异&#xff0c;简…

活动星投票技能创意大赛网络评选微信的投票方式线上免费投票

“技能创意大赛”网络评选投票_建立投票链接_作品投票小程序_扫码投票制作方法现在来说&#xff0c;公司、企业、学校更多的想借助短视频推广自己。通过微信投票小程序&#xff0c;网友们就可以通过手机拍视频上传视频参加活动&#xff0c;而短视频微信投票评选活动既可以给用户…

C语言详解——自定义类型:结构体、枚举、联合体

&#x1f307;个人主页&#xff1a;平凡的小苏 &#x1f4da;学习格言&#xff1a;别人可以拷贝我的模式&#xff0c;但不能拷贝我不断往前的激情 &#x1f6f8;C语言专栏&#xff1a;https://blog.csdn.net/vhhhbb/category_12174730.html 小苏希望大家能从这篇文章中收获到许…

【Pytorch】稀疏矩阵的表示与加减运算

前提知识&#xff1a;LongTensor()是64位整数型向量&#xff0c;FloatTensor()是32位浮点数向量。 有大量0元素的时候&#xff0c;我们可以使用坐标形式存储稀疏矩阵。 一个3*3的矩阵&#xff0c;但是只有坐标&#xff08;0,0&#xff09;处有值&#xff08;值为1&#xff09;…

量子计算(二十二):Grover算法

文章目录 Grover算法 一、什么是搜索算法 二、怎么实现Grover搜索算法 Grover算法 一、什么是搜索算法 举一个简单的例子&#xff0c;在下班的高峰期&#xff0c;要从公司回到家里&#xff0c;开车走怎样的路线才能够耗时最短呢&#xff1f;最简单的想法&#xff0c;当然…