【C】—文件版本通讯录的实现

news2025/7/8 18:28:21

关于C语言的知识放在专栏:C
小菜坤日常上传gitee代码:https://gitee.com/qi-dunyan
❤❤❤
个人简介:双一流非科班的一名小白,期待与各位大佬一起努力!
在这里插入图片描述

目录

  • 思路
  • 代码实现
  • 完整代码(可自取)

思路

在前面的文章中,已经讲解了动态版本的通讯录的实现,但是动态通讯录存在一个致命缺陷,就是它不能自动保存数据,而前面一篇文章中学到了数据持久化的方法之一:即把数据存放在磁盘文件上,便可以实现数据持久化。
具体应该如何做呢?
假如我们在退出的时候,通过文件操作,把我们所写的数据存在磁盘文件里,然后我们再进行下一次的使用的时候,在初始化阶段就从磁盘中读取这些数据,这不就实现了。

代码实现

代码的实现并不困难,只不过是在动态内存版本的基础上进行了一些文件操作,用来保存和读取数据。
从文件中读取信息(初始化阶段完成)

//读通讯录文件信息
//size_t fread(void* ptr, size_t size, size_t count, FILE* stream)
void Load_Contact(struct contact* p)
{
	//二进制读
	FILE* pfR = fopen("通讯录.txt", "rb");
	if (pfR == NULL)
	{
		//读取失败打印错误报告
		perror("Load_Contact::fopen");
		return;
	}
	struct message pf = { 0 };
	//fread返回值为读取的完整的元素个数,这里读取成功返回1,失败0
	while (fread(&pf, sizeof(pf), 1, pfR))
	{
		//判断是否增容
		check_capacity(p);
		p->data[p->sz] = pf;
		p->sz++;
	}
	//关闭文件
	fclose(pfR);
	pfR = NULL;
}

//初始化通讯录
void Init_contact(struct contact* p)
{
	assert(p);
	//开辟空间
	p->data =(struct message*) malloc(DEFAULT_SZ * sizeof(struct message));
	//假如开辟失败,报错
	if (p->data == NULL)
	{
		printf("%s\n", strerror(errno));
		return;
	}
	p->sz = 0;
	p->capacity = DEFAULT_SZ;
	//加载通讯录信息
	Load_Contact(p);
}

这里在动态版本的基础上,在初始化阶段加入了一个Load_Contact()函数,这个是用来以二进制读的方式打开文件,并且把读取到的信息放在结构体pf里,然后再将pf赋值到p指向的data数组的下标为size空间。
在这里插入图片描述
将数据写入文件(退出时保存信息)
在这里插入图片描述
这一步是为了将我们本次所写的数据,写入到文件中去,以备下一次打开时好从中读取数据。具体代码如下:

//写数据(保存通讯录信息)
//size_t fwrite(const void* ptr, size_t size, size_t count, FILE* stream)
void Storage_Contact(struct contact* p)
{
	//二进制形式写
	FILE* pfW = fopen("通讯录.txt", "wb");
	if (pfW == NULL)
	{
		//失败则打印错误报告
		perror("Storage_Contact::fopen");
		return;
	}
	int i = 0;
	//将sz个数据都写入到文件中
	for (i = 0; i < p->sz; i++)
	{
		fwrite(p->data + i, sizeof(struct message), 1, pfW);
	}
	//关闭
	fclose(pfW);
	pfW = NULL;
}

正是有了这一步,我们在退出时会创建一个文件来保存信息。供下一次读取。如以下视频所示:在我第二次运行程序时,上一次的数据已经加载完毕了。实现了数据持久化。

文件版本通讯录(退出可保存信息)

完整代码(可自取)

.h头文件

#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
#include<Windows.h>
#include<errno.h>

#define NAME 20
#define SEX 5
#define TELE 12
#define ADDR 30

#define DEFAULT_SZ 3//初始容量
#define INC_SZ 2//扩容

//联系人信息
struct message
{
	//姓名
	char name[NAME];
	//性别
	char sex[SEX];
	//电话
	char tele[TELE];
	//住址
	char addr[ADDR];
	//年龄
	int age;
};
//通讯录
struct contact
{
	struct message* data;
	int sz;//个数
	int capacity;//通讯录容量
};
//初始化通讯录
void Init_contact(struct contact* p);
//动态增加联系人
void Add_contact(struct contact* p);
//显示联系人
void Show_contact(const struct contact* p);
//删除联系人
void Dele_contact(struct contact* p);
//修改联系人信息
void revise_contact(struct contact* p);
//查找联系人信息
void Find_contact(const struct contact* p);
//排序联系人
void Sort_contact(struct contact* p);
//清空联系人
void Clean(struct contact* p);
//释放空间
void Destory_contact(struct contact* p);
//保存数据
void Storage_Contact(struct contact* p);

.c源文件(函数定义)

#include"Contact_exe.h"

//是否判断增容
int check_capacity(struct contact* p)
{
	//当联系人个数 == 通讯录容量时,增容INC_SZ个内存空间
	if (p->sz == p->capacity)
	{
		struct message* ptr = (struct message*)realloc(p->data, (p->capacity + INC_SZ) * sizeof(struct message));
		if (ptr == NULL)//判断是否增容失败
		{
			printf("%s\n", strerror(errno));
			return 0;
		}
		else
		{
			p->data = ptr;//增容成功,data就指向这块新开辟的空间
			p->capacity += INC_SZ;//容量+=INC_SZ
			//printf("增容成功!\n");
			return 1;
		}
	}
	//不需要增容
	else
		return 1;
}

//读通讯录文件信息
//size_t fread(void* ptr, size_t size, size_t count, FILE* stream)
void Load_Contact(struct contact* p)
{
	//二进制读
	FILE* pfR = fopen("通讯录.txt", "rb");
	if (pfR == NULL)
	{
		//读取失败打印错误报告
		perror("Load_Contact::fopen");
		return;
	}
	struct message pf = { 0 };
	//fread返回值为读取的完整的元素个数,这里读取成功返回1,失败0
	while (fread(&pf, sizeof(pf), 1, pfR))
	{
		//判断是否增容
		check_capacity(p);
		p->data[p->sz] = pf;
		p->sz++;
	}
	//关闭文件
	fclose(pfR);
	pfR = NULL;
}

//初始化通讯录
void Init_contact(struct contact* p)
{
	assert(p);
	//开辟空间
	p->data =(struct message*) malloc(DEFAULT_SZ * sizeof(struct message));
	//假如开辟失败,报错
	if (p->data == NULL)
	{
		printf("%s\n", strerror(errno));
		return;
	}
	p->sz = 0;
	p->capacity = DEFAULT_SZ;
	//加载通讯录信息
	Load_Contact(p);
}


//增加联系人
void Add_contact(struct contact* p)
{
	assert(p);
	if (0 == check_capacity(p))
	{
		printf("%s\n", strerror(errno));
		return;
	}
	printf("请输入姓名:->");
	scanf("%s", p->data[p->sz].name);

	printf("请输入性别:->");
	scanf("%s", p->data[p->sz].sex);

	printf("请输入电话:->");
	scanf("%s", p->data[p->sz].tele);

	printf("请输入住址:->");
	scanf("%s", p->data[p->sz].addr);

	printf("请输入年龄:->");
	scanf("%d", &(p->data[p->sz].age));
	system("cls");

	printf("增加成功!\n");
	printf("\n");
	p->sz++;
}
//显示联系人
void Show_contact(const struct contact* p)
{
	assert(p);
	int i = 0;
	printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5s\n", "姓名", "性别", "电话", "住址", "年龄");
	for (i = 0; i < p->sz; i++)
	{
		printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5d\n", p->data[i].name,
			p->data[i].sex,
			p->data[i].tele,
			p->data[i].addr,
			p->data[i].age);
	}
}
int find_name(const struct contact* p, char arr[])
{
	assert(p);
	int i = 0;
	for (i = 0; i < p->sz; i++)
	{
		if (0 == strcmp(p->data[i].name, arr))
			return i;
	}
	return -1;
}
//删除联系人
void Dele_contact(struct contact* p)
{
	assert(p);
	char del_name[NAME];
	printf("请输入要删除联系人的姓名:->");
	scanf("%s", del_name);
	//查找该联系人
	int ret = find_name(p, del_name);
	if (ret == -1)
		printf("查无此人!\n");
	else
	{
		int j = 0;
		for (j = ret; j < p->sz - 1; j++)
		{
			p->data[j] = p->data[j + 1];
		}
		p->sz--;
		system("cls");

		printf("删除成功!\n");
		printf("\n");
	}
}
//修改菜单栏
void menu_()
{
	printf("***********************************\n");
	printf("******   1、修改联系人姓名   ******\n");
	printf("******   2、修改联系人电话   ******\n");
	printf("******   3、修改联系人年龄   ******\n");
	printf("******   4、修改联系人住址   ******\n");
	printf("******   5、修改联系人性别   ******\n");
	printf("******   0、返回主菜单       ******\n");

}
//修改联系人信息
void revise_contact(struct contact* p)
{
	assert(p);
	char del_name[NAME];
	printf("请输入要修改信息的联系人的姓名:->");
	scanf("%s", del_name);
	int ret = find_name(p, del_name);
	if (ret == -1)
	{
		printf("查无此人!\n");
		printf("\n");
	}
	else
	{
		int in_put = 0;
		do
		{
			menu_();
			scanf("%d", &in_put);
			switch (in_put)
			{
			case 1:
				printf("请输入修改后的姓名:->");
				scanf("%s", p->data[ret].name);
				system("cls");

				printf("姓名修改成功!\n");
				break;
			case 2:
				printf("请输入修改后的电话:->");
				scanf("%s", p->data[ret].tele);
				system("cls");

				printf("电话修改成功!\n");
				break;
			case 3:
				printf("请输入修改后的年龄:->");
				scanf("%d", &(p->data[ret].age));
				system("cls");

				printf("年龄修改成功!\n");
				break;
			case 4:
				printf("请输入修改后的住址:->");
				scanf("%s", p->data[ret].addr);
				system("cls");

				printf("住址修改成功!\n");
				break;
			case 5:
				printf("请输入修改后的性别:->");
				scanf("%s", p->data[ret].sex);
				system("cls");

				printf("性别修改成功!\n");
				break;
			case 0:
				printf("取消修改!\n");
				break;
			default:
				printf("输入错误!\n");
				break;
			}
		} while (in_put);
	}
}
//查找联系人信息
void Find_contact(const struct contact* p)
{
	assert(p);
	char del_name[NAME];
	printf("请输入要查找联系人的姓名:->");
	scanf("%s", del_name);
	system("cls");

	//查找该联系人
	int ret = find_name(p, del_name);
	if (ret == -1)
		printf("查无此人!\n");
	else
	{
		printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5s\n", "姓名", "性别", "电话", "住址", "年龄");
		printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5d\n", p->data[ret].name,
			p->data[ret].sex,
			p->data[ret].tele,
			p->data[ret].addr,
			p->data[ret].age);
	}
}
//排序菜单
void menu_sort()
{
	printf("******   1、姓名   ******\n");
	printf("******   2、住址   ******\n");
	printf("******   3、年龄   ******\n");
	printf("******   4、性别   ******\n");
	printf("******   0、退出   ******\n");
}

//姓名排序
int cmp_name(const void* e1, const void* e2)
{
	return strcmp(((struct message*)e1)->name, ((struct message*)e2)->name);
}
//住址排序
int cmp_addr(const void* e1, const void* e2)
{
	return strcmp(((struct message*)e1)->addr, ((struct message*)e2)->addr);
}
//年龄排序
int cmp_age(const void* e1, const void* e2)
{
	return ((struct message*)e1)->age - ((struct message*)e2)->age;
}
//性别排序
int cmp_sex(const void* e1, const void* e2)
{
	return strcmp(((struct message*)e1)->sex, ((struct message*)e2)->sex);
}
//排序联系人
void Sort_contact(struct contact* p)
{
	int s = 0;
	do
	{
		//排序菜单
		menu_sort();
		printf("请选择排序类型:->");
		scanf("%d", &s);
		system("cls");
		switch (s)
		{
		case 1:
			qsort(p->data, p->sz, sizeof(struct message), cmp_name);
			printf("排序成功!\n");
			break;
		case 2:
			qsort(p->data, p->sz, sizeof(struct message), cmp_addr);
			printf("排序成功!\n");
			break;
		case 3:
			qsort(p->data, p->sz, sizeof(struct message), cmp_age);
			printf("排序成功!\n");
			break;
		case 4:
			qsort(p->data, p->sz, sizeof(struct message), cmp_sex);
			printf("排序成功!\n");
			break;
		case 0:
			printf("退出排序\n");
			break;
		default:
			printf("输入有误!\n");
			break;
		}

	} while (s);
}

//清空联系人
void Clean(struct contact* p)
{
	p->sz = 0;
	printf("清空成功!\n");
}

//释放空间
void Destory_contact(struct contact* p)
{
	free(p->data);
	p->data=NULL;
	p->sz = 0;
	p->capacity = 0;
}
//写数据(保存通讯录信息)
//size_t fwrite(const void* ptr, size_t size, size_t count, FILE* stream)
void Storage_Contact(struct contact* p)
{
	//二进制形式写
	FILE* pfW = fopen("通讯录.txt", "wb");
	if (pfW == NULL)
	{
		//失败则打印错误报告
		perror("Storage_Contact::fopen");
		return;
	}
	int i = 0;
	for (i = 0; i < p->sz; i++)
	{
		fwrite(p->data + i, sizeof(struct message), 1, pfW);
	}
	fclose(pfW);
	pfW = NULL;
}

.c源文件(测试)

#include"Contact_exe.h"

void menu()
{
	printf("--------------------------------------------------------------\n");
	printf("---------   1、增加联系人       2、删除指定联系人    ---------\n");
	printf("---------   3、修改联系人信息   4、查找联系人        ---------\n");
	printf("---------   5、排序联系人       6、显示已有联系人    ---------\n");
	printf("---------   0、退出并保存信息   7、清空联系人        ---------\n");
	printf("--------------------------------------------------------------\n");
}

int main()
{
	int input = 0;
	//创建通讯录
	struct contact con;
	//初始化通讯录(读之前通讯录的信息)
	Init_contact(&con);
	do
	{
		menu();
		printf("请选择:->");
		scanf("%d", &input);
		system("cls");
		switch (input)
		{
		case 1:
			//增加联系人
			Add_contact(&con);
			break;
		case 2:
			//删除联系人
			Show_contact(&con);
			Dele_contact(&con);
			break;
		case 3:
			//修改联系人信息
			Show_contact(&con);
			revise_contact(&con);
			break;
		case 4:
			//查找联系人信息
			Find_contact(&con);
			break;
		case 5:
			//排序联系人信息
			Show_contact(&con);
			Sort_contact(&con);
			break;
		case 6:
			//显示联系人
			system("cls");
			Show_contact(&con);
			printf("\n");
			break;
		case 7:
			//清空联系人
			Clean(&con);
			break;
		case 0:
			//保存数据
			Storage_Contact(&con);
			//释放空间
			Destory_contact(&con);
			break;
		default:
			printf("输入错误!\n");
			break;
		}
	} while (input);
	return 0;
}

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

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

相关文章

【SQL】索引的创建与设计原则

【SQL】索引的创建与设计原则索引的声明与使用索引的创建删除索引索引设计原则数据准备&#xff1a;哪些情况适合创建索引不适合创建索引的情况索引的声明与使用 索引的创建 # 索引的创建 #法1:create table # 隐式方式创建索引.在声明有主键约束、唯一性约束、外键约束的字段…

java项目-第146期ssm人事工资管理系统(spring+springmvc+mybatis+jsp)_java毕业设计_计算机毕业设计

java项目-第146期ssm人事工资管理系统(springspringmvcmybatisjsp)_java毕业设计_计算机毕业设计 【源码请到资源专栏下载】 今天分享的项目是《ssm人事工资管理系统》 该项目分为2个角色&#xff0c;管理员和用户。 用户可以浏览前台,包含功能有&#xff1a; 个人中心、公告信…

gin binding方法汇总

为了能够更方便的获取请求相关参数&#xff0c;提高开发效率&#xff0c;gin框架提供了Bind系列方法。 分类 Bind系列方法大致分两类&#xff1a; Should类型Must类型 二者的区别就是&#xff1a;should和must的区别&#xff0c;即must绑定失败会报错 关系图 Should类型 …

【licheePi-dock】驱动一个OLED-SSD1306

【licheePi-dock】驱动一个OLED-SSD1306OLED驱动修改设备树向内核添加驱动1.找到一个合适的位置&#xff0c;创建oled驱动文件&#xff1a;修改当前目录下的Kconfig和Makefile把驱动添加进内核编译编写应用程序编译我使用的是cmakeOLED驱动 本文使用的是i2c驱动的OLED【Lichee…

【PowerQuery】在Excel中手动刷新数据连接

将数据通过PowerQuery 导入进来后,这里将进行数据分组运算,最终的数据计算结果将保存在Excel 表格中,图1为数据导入结果。 图1 使用PowerQuery 加载导入的数据 在Excel中,如果我们希望进行销售统计的手动更新可以使用几种不同的方法来进行刷新这些数据。 1.刷新单一数据连…

UJNOJ_1000-1007_python

1000: AB Problem 内存限制&#xff1a;128 MB时间限制&#xff1a;10 S 评测方式&#xff1a;文本比较命题人&#xff1a;jiangyan 提交&#xff1a;109解决&#xff1a;87 提交提交记录统计 题目描述 Calculate ab result 输入 Two integer a,b (0<a,b<10) …

青少年python系列 46.文件操作2

青少年python教学视频ppt源码 青少年ptyhon可以参加的主流比赛大全_老程序员115的博客-CSDN博客 在上节课我们学习了如何查询文本信息内容&#xff0c;那我们能否使用Python来往文本中保存数据信息呢&#xff1f;当然也是可以的。Python可以完成将数据信息写入文件&#xff0c…

【机器学习】主成分分析

有任何的书写错误、排版错误、概念错误等&#xff0c;希望大家包含指正。 维数灾难 在高维情形下出现的数样本稀疏、距离计算困难等问题&#xff0c;是所有机器学习方法共同面临的的严重障碍&#xff0c;被称为“维数灾难”或“维数危机”&#xff08;curse of dimensionality…

MongoDB的安装配置及使用(WIndows/Java)

Hi&#xff0c;I‘m Shendi 下面主要介绍使用 Windows安装 MongoDB 和使用 Java 操作 文章目录MongoDB下载创建数据目录运行配置安装服务创建用户用户角色列表Java使用查询数据插入数据修改数据删除数据设置数据过期时间其他问题MongoClient是否需要关闭&#xff1f;内存为什么…

2022.11组队学习——跨模态视频搜索VCED

文章目录一、环境配置&#xff08;1.2章节可跳过&#xff09;1.1 docker启动本项目1.1.1 win10下安装docker1.1.2 配置容器1.1.3 启动项目1.2 使用vmware下的liunx启动本项目1.2.1 升级python3.91.2.2 安装其他包1.3 使用WSL下的linux启动本项目1.3.1 安装WSL1.3. 2安装Ubuntu1…

预测市场赛道新机遇:Moonquiz生态基于FIFA WORLD CUP推出首个预测应用

对未来事件的预测始终有着较为重要的意义&#xff0c;无论是对天气、地质灾害等的预测&#xff0c;还是对金融市场、事件走势等的预测&#xff0c;都有望让个人和组织对潜在的风险进行对冲&#xff0c;并获得最大化收益。 从Web3领域看&#xff0c;预测市场是活跃于加密行业早期…

M1 在VMware里安装Centos8

本文档主要介绍Mac M1芯片如何通过Vmware虚拟机&#xff0c;安装Centos系统。 安装准备 安装之前需要下载虚拟机安装包和Centos8镜像。 百度云自取链接: https://pan.baidu.com/s/19bPaU5Vf8HG6sv3C_jZDXw 提取码: 6jen 链接内容如下&#xff0c;centos8镜像&#xff0c;vm…

Linux零拷贝解析

目录 先备知识 物理内存 虚拟内存 内核空间和用户空间 内核空间 用户空间 DMA传输 上下文切换 传统IO执行流程 零拷贝实现的几种方式 用户态直接I/O mmapwrite实现的零拷贝 sendfile实现的零拷贝&#xff08;用户不可见&#xff09; sendfileDMA scatter/gather实…

kubernetes(K8S)学习笔记P1:基本概念和架构

kubernetes&#xff08;K8S&#xff09;学习笔记1.基本概念和架构1.1K8s概述和特性1.1.1概述1.1.2特性1.2K8s架构组件1.3平台规划1.4服务器硬件配置要求1.4.1测试环境1.4.2开发环境 (要求更高)1.5搭建k8s集群部署方式1.5.1 kubeadm 方式&#xff08;简单&#xff09;1.5.2二进制…

数据结构---串(整个部分)

串基本概念&#xff1a;串是由零个或者多个字符组成的有限序列&#xff0c;一半记作Sa1,a2,a3,a4.......&#xff08;n>0&#xff0c;串的长度&#xff09; 1.S 串的名字 n 串当中字符串的个数&#xff0c;称为串的长度。 串的常用术语 1.空串&#xff08;null stri…

[附源码]计算机毕业设计JAVA红河旅游信息服务系统

[附源码]计算机毕业设计JAVA红河旅游信息服务系统 项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM my…

Swin Transformer代码实现部分细节重点

swin transformer 1.patch-merging部分 代码&#xff1a;【amazing】 x0 x[:, 0::2, 0::2, :] # [B, H/2, W/2, C] 对应图片所有 1 的位置x1 x[:, 1::2, 0::2, :] # [B, H/2, W/2, C] 对应图片所有 3 的位置x2 x[:, 0::2, 1::2, :] # [B, H/2, W/2, C] 对应图片所有…

pve独显直连

目录折腾初步工作安装配置源去除订阅提示安装vim开启iommu屏蔽显卡驱动创建虚拟机体验折腾 买了个新笔记本&#xff0c;老的笔记本也没啥用了&#xff0c;挂二手平台也出不了多少钱。就想着自己折腾的新东西。之前有个PVE虚拟机感觉很不错&#xff0c;现在尝试一下怎么使用 初…

用HTML+CSS6音乐吧 7页

⛵ 源码获取 文末联系 ✈ Web前端开发技术 描述 网页设计题材&#xff0c;DIVCSS 布局制作,HTMLCSS网页设计期末课程大作业 | 音乐网页设计 | 仿网易云音乐 | 各大音乐官网网页 | 明星音乐演唱会主题 | 爵士乐音乐 | 民族音乐 | 等网站的设计与制作 | HTML期末大学生网页设计作…