【C语言】通讯录实现(下)

news2024/11/19 17:29:13

33c5ad031a694374825426e4107681f0.jpg

 

目录

 

1.进阶通讯录特点(下)

2.实现步骤

(1)保存增加的联系人数据到文件中

(2)加载保存的联系人数据

3.完整C语言通讯录代码

(1)contact.h

 (2)test.c

(3)contact.c

4.结语 


 

1.进阶通讯录特点(下)

①基本的增删查改功能;

②通讯录的空间不固定,大小可以调整;

③通过C语言文件操作来储存通讯录信息到硬盘上;

2.实现步骤

基本实现步骤同上、中两篇 【[C语言]通讯录实现(中) - CSDN App】http://t.csdnimg.cn/K5tqO

【【C语言】通讯录实现(上) - CSDN App】http://t.csdnimg.cn/ZrZQk

此外还有文件加载、写入、关闭等实现如下:

(1)保存增加的联系人数据到文件中

void SaveContact(Contact* pc)//保存联系人到文件中
{
	FILE* pf = fopen("contact.dat", "wb");//打开文件
	if (pf == NULL)//判断指针是否为空
	{
		perror("SaveContact");
		return;
	}
	//写数据
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		fwrite(pc->data+i, sizeof(PeoInfo), 1, pf);//一回写一个大小为PeoInfo的文件
	//从pc->data+i的位置开始读写到文件中
	}
	//关闭文件
	fclose(pf);
	pf = NULL;
}

运行后得到一个contact.dat的文件用记事本打开如图:

0296e054f4384527acc14b0c4ae65ae6.png

因为fopen函数的打开方式为“wb”,二进制写入模式所以打开该文件是一堆乱码 

3402a3e42fa34df3ba67e54a283b10c2.png

如上图所示生成了contact.dat文件

(2)加载保存的联系人数据

void LoadContact(Contact* pc)//加载已保存的联系人数据
{
	//打开文件
	FILE* pf = fopen("contact.dat", "rb");
	//以二进制形式读文件
	PeoInfo tmp = { 0 };//创建一个变量来存放读到的数据
	while (fread(&tmp, sizeof(PeoInfo), 1, pf))//while循环一直读到文件末尾没有数据为止
	{
		if (CheckCapacity(pc) == 0)           //要先检查当前通讯录容量是否够存放保存的联系人
		{
			printf("无法加载联系人哦~\n");   //CheckCapacity为0时表示无法增容可能没办法开辟空间
			return;
		}
		pc->data[pc->sz] = tmp;          //将保存的联系人写入当前通讯录
		pc->sz++;                       //写入一个sz通讯录联系人数量加一和增加联系人函数一样
	}
	printf("已保存的联系人加载成功啦~\n");
	return;
}

3.完整C语言通讯录代码

(1)contact.h

#pragma once
//定义的的头文件contact.h
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30

#define DEFAULT_SZ 3
#define INT_SZ 2
enum OPTION//用枚举来定义变量
{
	EXIT,//0
	ADD,//1
	DEL,//2...
	SEARCH,
	MODIFY,
	SHOW,
	SORT
};


//声明类型
typedef struct PeoInfo //创建一个结构体来储存联系人相关信息
{
	char name[MAX_NAME];
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
	int age;
}PeoInfo;

//通讯录
typedef struct Contact
{
	PeoInfo* data;//创建一个PeoInfo指针来指向存放数据的空间
	int sz;//用来记录联系人的个数
	int capacity;//记录通讯录当前最大容量
}Contact;

//函数声明
void InitContact(Contact* pc);
void AddContact(Contact* pc);
void ShowContact(const Contact* pc);
void DelContact(Contact* pc);
void SearchContact(const Contact* pc);
void DestroyContact(Contact* pc);
void SaveContact(Contact* pc);
void LoadContact(Contact* pc);//加载已保存的联系人数据





 (2)test.c

#define _CRT_SECURE_NO_WARNINGS 1
//test.c文件--流程
#include"contact.h"
void menu()//菜单
{
	printf("*******Contact******\n");
	printf("*** 1.ADD  2.DEL ***\n");
	printf("***** 3.SEARCH *****\n");
	printf("***** 4.MODIFY *****\n");
	printf("** 5.SHOE  6.SORT **\n");
	printf("****** 0.EXIT ******\n");

}
int main()//主函数
{
	int sec = 1;
	Contact con;
	InitContact(&con);//初始化通讯录
	LoadContact(&con);

	while (sec)
	{
		menu();
		scanf("%d", &sec);
		switch (sec)//选择不同的功能
		{
		case EXIT://退出
			SaveContact(&con);
			DestroyContact(&con);
			printf("您已退出\n");
			break;
		case ADD://加
			AddContact(&con);
			break;
		case DEL://删
			DelContact(&con);
			break;
		case SEARCH://查
			SearchContact(&con);
			break;
		case MODIFY://改
			break;
		case SHOW://显示
			ShowContact(&con);
			break;
		case SORT://分类
			break;
		default:
			printf("选择错误,请重新输入\n");
		}
		printf("\n");
	}
	return 0;
}

(3)contact.c

#define _CRT_SECURE_NO_WARNINGS 1
//contact函数实现contact.c源文件
#include"contact.h"

 int CheckCapacity(Contact* pc);
void LoadContact(Contact* pc)//加载已保存的联系人数据
{
	//打开文件
	FILE* pf = fopen("contact.dat", "rb");
	//读文件
	PeoInfo tmp = { 0 };
	while (fread(&tmp, sizeof(PeoInfo), 1, pf))
	{
		if (CheckCapacity(pc) == 0)
		{
			printf("无法加载联系人哦~\n");
			return;
		}
		pc->data[pc->sz] = tmp;
		pc->sz++;
	}
	printf("已保存的联系人加载成功啦~\n");
	return;
}
int Search_by_name(const Contact* pc)//查找函数
{

	char name[MAX_NAME] = { 0 };
	scanf("%s",name);
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(pc->data[i].name, name) == 0)
		{
			return i;
		}
	}
	return -1;
}


void InitContact(Contact* pc) //初始化通讯录函数
{
	//memset(pc->data, 0, sizeof(pc->data));//静态版本
	assert(pc);
	pc->data = malloc(DEFAULT_SZ * sizeof(PeoInfo));//开辟空间
	if (pc->data == NULL)//判断指针是否为空
	{
		perror("InitContact");
		return;
	}
	pc->capacity = DEFAULT_SZ;//使用了宏定义DEFAULT_SZ=3容量
	pc->sz = 0;
}

int CheckCapacity(Contact* pc)//增容函数 加联系人时如果空间不够就增容
{
	if (pc->sz == pc->capacity)
	{
		PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INT_SZ)*sizeof(PeoInfo));
		//如果容量满了用realloc函数增容,INT_SZ宏定义为2
		if (ptr == NULL)
		{
			perror("CheckCapacity");
			return 0;
		}
		else
		{
			pc->data = ptr;//将增容后的指针赋给pc->data
			pc->capacity += INT_SZ;//capacity相应增加
			printf("增容成功\n");
		}
		return 1;
	}
	return 1;
}


void AddContact(Contact* pc)//增加联系人
{
	assert(pc);
	/*if (pc->sz == MAX)
	{
		printf("联系人已满\n");
		return;
	}*/
	if (0 == CheckCapacity(pc))
	{
		printf("无法增容哦~\n");
		return;
	}
	printf("请输入要添加的联系人名字、性别、年龄、电话、地址:\n");
	scanf("%s %s %d %s %s",pc->data[pc->sz].name, pc->data[pc->sz].sex,&pc->data[pc->sz].age, pc->data[pc->sz].tele,pc->data[pc->sz].addr);
	printf("您已经成功添加%s\n", pc->data[pc->sz].name);
	pc->sz++;
	return;
}

void ShowContact(const Contact* pc)//显示联系人
{
	assert(pc);
	if (pc->sz == 0)
	{
		printf("您还未添加联系人哦~快选择1去添加吧~\n");
		return;
	}
	int i = 0;
	printf("%-10s\t%-5s\t%-5s\t%-15s\t%-30s\t\n", "名字", "性别", "年龄", "电话", "住址");
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-10s\t%-5s\t%-5d\t%-15s\t%-30s\t\n",
			pc->data[i].name, pc->data[i].sex,
			pc->data[i].age, pc->data[i].tele,
			pc->data[i].addr);
	}
	return;
}

void DelContact(Contact* pc)//删除联系人
{
	assert(pc);
	if (pc->sz == 0)
	{
		printf("您还没有加联系人哦~快选择1去添加吧~\n");
		return;
	}
	printf("请输入要删除的联系人的名字:\n");
	int flag = Search_by_name(pc);
	if (flag == -1)
	{
		printf("没有找到该联系人哦~\n");
		return;
	}
	int j = 0;
	for (j = flag; j < pc->sz - 1; j++)
	{
		pc->data[j] = pc->data[j + 1];
	}
	printf("您已经成功删除该联系人\n");
	pc->sz--;
	return;
}

void SearchContact(const Contact* pc)//查找联系人并打印
{
	printf("请输入要查找的联系人的名字:\n");
	int i = Search_by_name(pc);
	if (i == -1)
	{
		printf("没有找到该联系人哦~\n");
		return;
	}
	else
	{
		printf("%-10s\t%-5s\t%-5d\t%-15s\t%-30s\t\n",
			pc->data[i].name, pc->data[i].sex,
			pc->data[i].age, pc->data[i].tele,
			pc->data[i].addr);
	}
	return;
}

//释放空间
void DestroyContact(Contact* pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->capacity = 0;
	pc->sz = 0;
}

void SaveContact(Contact* pc)//保存增加的联系人到文件中
{
	FILE* pf = fopen("contact.dat", "wb");//打开文件
	if (pf == NULL)//判断指针是否为空
	{
		perror("SaveContact");
		return;
	}
	//写数据
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		fwrite(pc->data+i, sizeof(PeoInfo), 1, pf);//一回写一个大小为PeoInfo的文件
	//从pc->data+i的位置开始读写到文件中
	}
	//关闭文件
	fclose(pf);
	pf = NULL;
}

4.结语 

三版通讯录循序渐进,从最开始的定容增删查改等功能的实现(上篇),到可以不定容的通讯录(中篇),最后到可以保存数据到文件并从文件中加载保存的联系人(下篇),我们不仅学会了一些基本函数的写法,还学了动态内存函数malloc,realloc等函数的用法,并熟悉了c语言文件的基本操作。

 

 

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

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

相关文章

【C++初阶】--入门基础(二)

目录 一.C输出与输入 二.缺省参数 1.概念 2.缺省参数分类 (1) 全缺省参数 (2)半缺省参数 三.函数重载 1.概念 2.C支持函数重载的原理--名字修饰 四.引用 1.概念 2.语法 3.引用的特性 (1)引用在定义时必须初始化 (2)引用时不能改变指向 (3)一个变量…

故障诊断 | 一文解决,RF随机森林的故障诊断(Matlab)

效果一览 文章概述 故障诊断 | 一文解决,RF随机森林的故障诊断(Matlab) 模型描述 随机森林(Random Forest)是一种集成学习(Ensemble Learning)方法,常用于解决分类和回归问题。它由多个决策树组成,每个决策树都独立地对数据进行训练,并且最终的预测结果是由所有决策…

使用apifox创建一个Mock Server Api 接口

安装 下载 Apifox - API 文档、调试、Mock、测试一体化协作平台。拥有接口文档管理、接口调试、Mock、自动化测试等功能&#xff0c;接口开发、测试、联调效率&#xff0c;提升 10 倍。最好用的接口文档管理工具&#xff0c;接口自动化测试工具。 创建mock api项目中使用 创建项…

【C++初阶】--入门基础(一)

目录 一.命名空间 1.前言引入 2.namespace关键字 (1)前言 (2)域作用限定符 :: (3)命名空间域namespace ① 细节理解 ② 命名空间的名称相同 ③命名空间的嵌套 (4)命名空间的使用 ① 加命名空间名称及作用域限定符 ②使用using将命名空间中某个成员引入 ③ 使用u…

【2024美赛】F题(中英文):减少非法野生动物贸易Problem F: Reducing Illegal Wildlife Trade

【2024美赛】F题&#xff08;中英文&#xff09;&#xff1a;减少非法野生动物贸易Problem F: Reducing Illegal Wildlife Trade 写在最前面2024美赛翻译 —— 跳转链接 中文赛题问题F&#xff1a;减少非法野生动物贸易你的工作应探讨以下子问题&#xff1a;参考文献词汇表 英文…

zabbix-监控应用程序(Mysql、Nginx)

基础环境&#xff1a;zabbix服务端、两台zabbix被监控端、分别安装Mysql和Nginx环境拓扑图&#xff1a; 实验目标&#xff1a;可以通过zabbix监控到被监控端上安装的mysql与nginx&#xff0c;通过zabbix实时检测mysql和nginx的数据。实验步骤&#xff1a;1.在8.7服务器上安装My…

推荐收藏!算法工程师面试常考的手撕面试题!

今天给大家分享一些算法工程师技术面试中常手撕的代码。 不管是秋招还是社招&#xff0c;互联网大厂的技术面试中的手撕代码这一部分总是绕不过去的一关。 如果你对这些感兴趣&#xff0c;可以文末找我们交流 手撕 numpy写线性回归的随机梯度下降&#xff08;stochastic gra…

实习|基于SSM的实习管理系统设计与实现(源码+数据库+文档)

实习管理系统目录 目录 基于SSM的实习管理系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、管理员功能介绍 &#xff08;1&#xff09;管理员登录 &#xff08;2&#xff09;实训方向管理 &#xff08;3&#xff09;公告信息管理 &#xff08;4&#xff0…

基于布谷鸟搜索的多目标优化matlab仿真

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.本算法原理 1. 布谷鸟搜索算法基础 2. 多目标优化问题 3. 基于布谷鸟搜索的多目标优化算法 4. 解的存储和选择策略 5.算法步骤 5.完整程序 1.程序功能描述 基于布谷鸟搜索的多目标优化&#xff0c;…

互补滤波算法介绍+SCL源代码(收放卷线速度处理)

工程上对测量信号进行处理,我们可以利用低通滤波器,还可以利用滑动平均值滤波等,关于低通滤波器和滑动平均值滤波器,可以参考专栏相关文章,常用链接如下: 博途PLC一阶滞后低通滤波器(支持采样频率设置) https://rxxw-control.blog.csdn.net/article/details/132972093h…

《统计学习方法:李航》笔记 从原理到实现(基于python)-- 第5章 决策树

文章目录 第5章 决策树5.1 决策树模型与学习5.1.1 决策树模型5.1.2 决策树与if-then规则5.1.3 决策树与条件概率分布5.1.4 决策树学习5.2 特征选择5.2.1 特征选择问题5.2.2 信息增益5.2.3 信息增益比5.3.1 ID3算法5.3.2 C4.5的生成算法5.4 决策树的剪枝5.5 CART算法5.5.1 CART生…

协作办公开源神器:ONLYOFFICE

目录 前言ONLYOFFICE为什么选择ONLYOFFICE强大的文档编辑功能多种协作方式多人在线协同支持跨端多平台连接器安全性极高本地部署 ONLYOFFICE 8.0版本震撼来袭可填写的 PDF 表单显示协作用户头像更新插件界面设计更快更强大 总结 前言 近几年来&#xff0c;随着互联网技术的不断…

【疑问】为什么声明和定义要分离

前言 我们在学习的时候接触过一个话&#xff1a;编写自定义函数的时候 要做到声明和定义分离 那么为什么呢 今天就来简单的了解一下 疑问&#xff1a;错误的发生 下面给出两个源文件和一个头文件以及报错信息 text.cc文件 #include"head.h"int main() {int a 1…

《熬夜整理》保姆级系列教程-玩转Wireshark抓包神器教程(1)-初识Wireshark

1.简介 前边已经介绍过两款抓包工具&#xff0c;应该是够用了&#xff0c;也能够处理在日常工作中遇到的问题了&#xff0c;但是还是有人留言让宏哥要讲解讲解Wireshark这一款抓包工具&#xff0c;说实话宏哥之前也没有用过这款工具&#xff0c;只能边研究边分享。换句话说就是…

Unity DOTween插件常用方法(二)

文章目录 1.3 动画设置1.4 动画队列 Sequence1.5 动画回调函数1.6 等待函数&#xff08;协程中使用&#xff09; 1.3 动画设置 SetLoops 设置循环动画&#xff1b; 参数&#xff1a; loops&#xff1a;指定循环的次数&#xff0c;设置为 -1 表示无限循环&#xff1b; loopType…

linux -- 并发 -- 并发来源与简单的解决并发的手段

互斥与同步 当多个执行路径并发执行时&#xff0c;确保对共享资源的访问安全是驱动程序员不得不面对的问题 互斥&#xff1a;对资源的排他性访问 同步&#xff1a;对进程执行的先后顺序做出妥善的安排 一些概念&#xff1a; 临界区&#xff1a;对共享的资源进行访问的代码片段…

年轻人“过年”行为大赏

【潮汐商业评论/原创】 前脚刚进家门&#xff0c;后脚快递电话一个接着一个。 临近春节&#xff0c;Julia是提前批回家的年轻人&#xff0c;与Julia一同到家的还有她的年货。上至大小家电&#xff0c;下到坚果零食&#xff0c;短短几天快递就堆满了客厅。 吃的喝的还能理解&…

Tencent Tinker:移动应用热修复的未来之路

Tencent Tinker&#xff1a;移动应用热修复的未来之路 1 引言 移动应用热修复是一项在移动应用开发领域中日益重要的技术&#xff0c;它可以帮助应用程序开发者快速修复线上应用的bug、漏洞和功能问题&#xff0c;而无需重新发布整个应用。这种能力对于提高用户体验、降低用户…

卸载Ubuntu双系统

卸载Ubuntu双系统 我们卸载Ubuntu双系统&#xff0c;可能出于以下原因&#xff1a; 1、Ubuntu系统内核损坏无法正常进入 2、Ubuntu系统分配空间不足&#xff0c;直接扩区较为复杂 3、以后不再使用Ubuntu&#xff0c;清理留出空间 123无论出于哪种原因&#xff0c;我们都是要…

vue项目在public中编写一个json文件 并用http请求获取 模仿数据接口

我们前后端分离是经常存在 前端已经开发到需要接口 但后端还没开始的情况的 如果直接在js中写假数据 后面还要改不少东西 多少有点麻烦 我们可以 直接在 public 静态资源目录下创建一个json文件 这里 我取名叫 city.json 大家可以根据自己喜好给json文件命名 我这个json文件的…