运用动态内存实现通讯录(增删查改+排序)
目录
前言:
实现通讯录:
1.创建和调用菜单:
2.创建联系人信息和通讯录:
3.初始化通讯录:
4.增加联系人:
5.显示联系人:
6.删除联系人:
编辑
7.查找联系人:
编辑
8.修改联系人:
编辑
9.排序联系人:
编辑
10.释放通讯录
总结:
前言:
通讯录通常是一个记录联系人信息的电子或纸质文件,包括名称、电话号码、电子邮件地址、物理地址等。通讯录旨在方便人们在需要联系某个人或组织时快速找到相关信息。现代通讯录通常是数字化的,可以存储在计算机、智能手机或云服务器中,也可以在社交媒体等在线平台上创建。通讯录是现代社交和商务通信的重要工具之一,有助于帮助人们管理他们的联系人,从而更好地进行社交和商务交流。
以下是一个简单的C语言实现通讯录的例子:
#include <stdio.h>
#include <string.h>#define MAX_CONTACTS 100 // 最大联系人数struct Contact {char name[50];char phone_num[20];char email[50];
}; // 联系人结构体int main() {struct Contact contacts[MAX_CONTACTS]; // 联系人数组int num_contacts = 0; // 当前联系人数while (1) {printf("请选择操作:\n");printf("1. 添加联系人\n");printf("2. 显示所有联系人\n");printf("3. 退出\n");int action;scanf("%d", &action);if (action == 1) { // 添加联系人if (num_contacts == MAX_CONTACTS) {printf("联系人数量已达上限\n");} else {struct Contact new_contact;printf("请输入联系人姓名:\n");scanf("%s", new_contact.name);printf("请输入联系人电话号码:\n");scanf("%s", new_contact.phone_num);printf("请输入联系人电子邮件地址:\n");scanf("%s", new_contact.email);contacts[num_contacts] = new_contact;num_contacts++;printf("联系人添加成功\n");}} else if (action == 2) { // 显示所有联系人printf("当前联系人如下:\n");printf("姓名\t电话\t邮箱\n");for (int i = 0; i < num_contacts; i++) {printf("%s\t%s\t%s\n", contacts[i].name, contacts[i].phone_num, contacts[i].email);}} else if (action == 3) { // 退出printf("程序已退出\n");break;} else {printf("输入无效,请重新输入\n");}}return 0;
}
这个程序使用一个结构体数组来存储所有联系人的信息。通过一个循环菜单来实现添加联系人、显示所有联系人和退出等操作。程序可以根据需要进行修改和扩展。
以上是最基础最基本的通讯录实现,
而本文则会运用动态内存对通讯录实现改进操作。
实现通讯录:
1.创建和调用菜单:
我们在之前的blog中都对菜单进行了创建,虽然说之前都是关于实现游戏的菜单,但这次也不例外。
通讯录也应当拥有一个菜单。
具体的代码我不多做赘述,如下:
void menu()
{printf("**************Contact**************\n");printf("***********************************\n");printf("*******1.ADD 2.DEL*********\n");printf("*******3.SEARCH 4.MODIFY******\n");printf("*******5.SHOW 6.SORT********\n");printf("*************0.EXIT****************\n");printf("***********************************\n");
}
接下来就是我们的do...while循环和switch语句的实现了,以上菜单代码在我的之前三篇blog中都有讲解,需要可以跳转到:
C语言实现《扫雷》_无双@的博客-CSDN博客
C语言实现《三子棋》游戏-CSDN博客
C语言实现《猜数字游戏》_无双@的博客-CSDN博客
如今我们学习了结构体的相关知识,并认识了有关枚举的结构体类型,我们不妨尝试一下:
enum option
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};
则我们在main函数里就可以写成:
int main()
{int input = 0;do{menu();printf("请选择:>");scanf("%d", &input);switch (input){case ADD:break;case DEL:break;case SEARCH:break;case MODIFY:break;case SHOW:break;case SORT:break;case EXIT:printf("正在退出...\n");break;default:printf("输入错误...\n");}} while (input);return 0;
}
运用枚举的好处就是,我们只要想调用ADD就直接输入1,不管ADD是case1里还是case2里。
如此则可以方便我们的操作,在以后的代码的实现,我们应当尽量尝试去使用枚举来实现菜单。
2.创建联系人信息和通讯录:
我们在创建联系人信息和通讯录的时候,是面向联系人和通讯录这两个对象的,因此我们不妨使用结构体来创建。
代码如下:
#define NAME_MAX 20
#define SEX_MAX 5
#define TELE_MAX 13
#define ADDR_MAX 40typedef struct PeoInfo
{char name[NAME_MAX];int age;char sex[SEX_MAX];char tele[TELE_MAX];char addr[ADDR_MAX];
}PeoInfo;typedef struct Contact
{PeoInfo* data;int sz;//记录当前联系人的个数int capacity;//记录当前通讯录的容量
}Contact;
联系人里应当有“名字”,“年龄”,“性别”,“电话”,“地址”。
通讯录里应当有“指向联系人的指针”,“记录当前联系人的个数”,“记录当前通讯录的容量”。
指针和容量是为之后动态开辟内存做好准备。
它们之间的关系如图:

3.初始化通讯录:
在完成上述的操作后,我们则可以开始对我们的通讯录进行操作。
第一步肯定得是初始化通讯录,
在这一步则是我们运用动态内存开辟空间的最佳时机。
代码如下:
#define DEFAULT_SZ 3int main()
{Contact con;InitContact(&con);
}void InitContact(Contact* pc)
{assert(pc);pc->sz = 0;pc->capacity = DEFAULT_SZ;pc->data = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));if (pc->data == NULL){perror("calloc->InitContact");return;}
}
由于我们之前已经讲解过结构体传参,对于结构体传参的最佳办法是传递地址,所以我们运用指针来进行操作。
我们使用calloc开辟空间,是因为使用calloc就可以帮我们对开辟好的空间直接初始化为0,这样可大大节省代码量,并且使得代码更为整洁。
pc->capacity = 3的意思是让空间容量在初始化的时候最多可以放下三个联系人,如果不够了我们就继续增加,使用动态内存开辟空间的优势就在这里:
方便我们进行追加联系人。
4.增加联系人:
在我们增加联系人的时候,我们首先需要判断我们开辟好的空间是否够用,这个时候我们应当在contact.c的文件中创建一个函数,用来检查空间是否够用。
函数代码如下:
#define DEFAULT_INC 2
static void Check_Capacity(Contact* pc)
{assert(pc);if (pc->sz == pc->capacity){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (DEFAULT_INC + DEFAULT_SZ)*sizeof(PeoInfo));if (ptr != NULL){pc->data = ptr;pc->capacity += DEFAULT_INC;printf("增容成功!\n");}else{perror("realloc->AddContact");return;}}
}
这里要注意的是我们在使用realloc追加空间的时候,应当创建一个临时指针,先用于判断realloc是否可以开辟成功,如果可以则赋值到pc->data处,这样可以使得代码风格更加健壮。
但我们判断完后,就要对代码进行添加操作,代码如下:
void AddContact(Contact* pc)
{assert(pc);Check_Capacity(pc);printf("请输入名字:>");scanf("%s", pc->data[pc->sz].name);printf("请输入年龄:>");scanf("%d", &(pc->data[pc->sz].age));printf("请输入性别:>");scanf("%s", pc->data[pc->sz].sex);printf("请输入电话:>");scanf("%s", pc->data[pc->sz].tele);printf("请输入地址:>");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("添加成功!\n");
}
5.显示联系人:
我们在实现增加联系人后,可以将已经存在的联系人信息打印出来,方便我们查看。
具体的代码如下:
void ShowContact(Contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空!\n");return;}printf("%-20s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");for (int i = 0; i < pc->sz; i++){printf("%-20s%-5d%-5s%-12s%-30s\n",pc->data[i].name,pc->data[i].age,pc->data[i].sex,pc->data[i].tele,pc->data[i].addr);}
}
我们想要打印出来的格式较为整齐,
所以采取使用
printf("%-20s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");
这样我们的输出结果就为这样:

6.删除联系人:
对于删除练习人,我们要进行的第一步操作当然需要找到联系人。
所以我们可以尝试创建一个函数用来查找联系人:
代码如图所示:
static int FindByName(Contact* pc, char* name)
{assert(pc && name);for (int i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name,name) == 0){return i;}}//找不到return -1;
}
name数组是在DelContact函数里创建的,用来输入名字。
这里我们运用了字符串比较函数,strcmp,如果它们相等则会等于0,,就说明找到了该联系人,则返回通讯录里的第i个联系人。
则我们在DelContact就可以进行删除操作。
具体的方法就是将后面的联系人一个一个与前一个进行替换。
代码如下:
for (int i = ret; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("该联系人已删除\n");
如此一来删除联系人代码完成,
完整代码如下:
void DelContact(Contact* pc)
{assert(pc);char name[NAME_MAX];if (pc->sz == 0){printf("通讯录为空\n");return;}printf("请输入你要删除联系人的名字:>");scanf("%s", name);int ret = FindByName(pc, name);if (ret == -1){printf("该联系人不存在\n");return;}for (int i = ret; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("该联系人已删除\n");
}
7.查找联系人:
查找联系人与上述相似,先查找名字,再进行输出,输出则用到的是ShowContact里的代码:
在这里我不做过的赘述,完整代码如下:
void SearchContact(Contact* pc)
{assert(pc);char name[NAME_MAX];if (pc->sz == 0){printf("通讯录为空!\n");return;}printf("请输入你要查找联系人的名字:>");scanf("%s", name);int ret = FindByName(pc, name);if (ret == -1){printf("不存在该联系人\n");return;}printf("%-20s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("%-20s%-5d%-5s%-12s%-30s\n",pc->data[ret].name,pc->data[ret].age,pc->data[ret].sex,pc->data[ret].tele,pc->data[ret].addr);
}
8.修改联系人:
修改联系人也和上述相似,先查找再进行修改,这次的修改则是运用了AddContact函数里的部分代码,
完整代码如下:
void ModifyContact(Contact* pc)
{assert(pc);char name[NAME_MAX];if (pc->sz == 0){printf("通讯录为空\n");return;}printf("请输入你要修改联系人的名字:>");scanf("%s", name);int ret = FindByName(pc, name);if (ret == -1){printf("不存在该联系人\n");return;}printf("请输入名字:>");scanf("%s", pc->data[ret].name);printf("请输入年龄:>");scanf("%d", &(pc->data[ret].age));printf("请输入性别:>");scanf("%s", pc->data[ret].sex);printf("请输入电话:>");scanf("%s", pc->data[ret].tele);printf("请输入地址:>");scanf("%s", pc->data[ret].addr);printf("修改成功!\n");
}
9.排序联系人:
对联系人的排序我们可以运用qsort函数来进行排序,如果忘记了该函数可以参考之前blog中对qsort函数的讲解:
自主实现qsort函数-CSDN博客
接下来我们则可以实现:
1.按照名字大小来排序。
2.按照年龄大小来排序。
代码如下:
static int cmp(const void* e1, const void* e2)
{//return (((PeoInfo*)e1)->age > ((PeoInfo*)e2)->age) ? 1 : -1;//年龄排序return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);//姓名排序
}void SortContact(Contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空!\n");return;}qsort(pc->data, pc->sz, sizeof(PeoInfo), cmp);printf("排序成功!\n");
}
10.释放通讯录
如今我们想要离开通讯录时,因为使用了动态内存的方式开辟内存,就不得不对其进行释放。
对于释放的操作,我们则可以创建一个DestoryContact函数来进行释放,以及销毁。
具体方式如下:
void DestoryContact(Contact* pc)
{assert(pc);free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;
}
如此一来通讯录操作完成。
总结:
本文实现了动态内存开辟实现通讯录。
该通讯录还存在一些问题,例如无法保存该数据,随着程序的结束,通讯录内容也就此结束。
因此我们下一次可以尝试使用文件操作来编写通讯录。
一下是我的Gitee仓库可以参考以上代码:
test_c_with_X1: 本仓库里的代码为c语言的测试代码 - Gitee.com
学习完后可以动手尝试编写编写。
记住
“坐而言不如起而行”
Action speaker louder than words!
相关文章:
运用动态内存实现通讯录(增删查改+排序)
目录 前言: 实现通讯录: 1.创建和调用菜单: 2.创建联系人信息和通讯录: 3.初始化通讯录: 4.增加联系人: 5.显示联系人: 6.删除联系人: 编辑 7.查找联系人: …...
基于Cplex的人员排班问题建模求解(JavaAPI)
使用Java调用Cplex实现了阿里mindopt求解器的案例(https://opt.aliyun.com/platform/case)人员排班问题。 这里写目录标题 人员排班问题问题描述数学建模编程求解(CplexJavaAPI)求解结果 人员排班问题 随着现在产业的发展&#…...
理解Go中的数据类型
引言 数据类型指定了编写程序时特定变量存储的值的类型。数据类型还决定了可以对数据执行哪些操作。 在本文中,我们将介绍Go的重要数据类型。这不是对数据类型的详尽研究,但将帮助您熟悉Go中可用的选项。理解一些基本的数据类型能让你写出更清晰、性能…...
【人工智能导论】线性回归模型
一、线性回归模型概述 线性回归是利用函数对一个或多个自变量和因变量之间关系进行建模的一种回归分析。简单来说,就是试图找到自变量与因变量之间的关系。 二、线性回归案例:房价预测 1、案例分析 问题:现在要预测140平方的房屋的价格&…...
十大常见排序算法详解(附Java代码实现和代码解析)
文章目录 十大排序算法⛅前言🌱1、排序概述🌴2、排序的实现🌵2.1 插入排序🐳2.1.1 直接插入排序算法介绍算法实现 🐳2.1.2 希尔排序算法介绍算法实现 🌵2.2 选择排序🐳2.2.1 选择排序算法介绍算…...
在Ubuntu上通过Portainer部署微服务项目
这篇文章主要记录自己在ubuntu上部署自己的微服务应用的过程,文章中使用了docker、docker-compose和portainer,在部署过程中遇到了不少问题,因为博主也是初学docker-compose,通过这次部署实战确实有所收获,在这篇文章一…...
软件测试基础学习
注意: 各位同学们,今年本人求职目前遇到的情况大体是这样了,开发太卷,学历高的话优势非常的大,公司会根据实际情况考虑是否值得培养(哪怕技术差一点);学历稍微低一些但是技术熟练的…...
移动手机截图,读取图片尺寸
这个代码的设计初衷是为了解决图片处理过程中的一些痛点。想象一下,我们都曾遇到过这样的情况:相机拍摄出来的照片、网络下载的图片,尺寸五花八门,大小不一。而我们又渴望将它们整理成一套拥有统一尺寸的图片,让它们更…...
服务器应用程序不可用的原因是什么引起的
服务器应用程序不可用的原因是什么引起的 服务器应用程序不可用的原因是什么引起的?其实服务器应用程序不可用可能是由多种原因引起的。主要包括软件故障、网络问题、硬件故障、安全问题、配置错误、容量不足、数据库问题等,具体详细服务器应用程序不可用的原因如下…...
使用SPY++查看窗口信息去排查客户端UI软件问题
目录 1、使用SPY++查看窗口的信息 2、使用SPY++查看某些软件UI窗口用什么UI组件实现的...
Flink CDC MySQL同步MySQL错误记录
1、启动 Flink SQL [appuserwhtpjfscpt01 flink-1.17.1]$ bin/sql-client.sh2、新建源表 问题1:Encountered “(” 处理方法:去掉int(11),改为int Flink SQL> CREATE TABLE t_user ( > uid int(11) NOT NULL AUTO_INCREMENT COMME…...
深入了解 Linux 中的 AWK 命令:文本处理的瑞士军刀
简介 在Linux和Unix操作系统中,文本处理是一个常见的任务。AWK命令是一个强大的文本处理工具,专门进行文本截取和分析,它允许你在文本文件中查找、过滤、处理和格式化数据。本文将深入介绍Linux中的AWK命令,让你了解其基本用法和…...
【RuoYi项目分析】网关的AuthFilter完成“认证”,注意是认证而不是权限
文章目录 1. 功能介绍2. AuthFilter的配置3. AuthFilter实现分析4. 资料参考 过滤器的功能是检验经过网关的每一个请求,检查 token 中的信息是否有效。 注意是“认证检查”,而不是“权限” 1. 功能介绍 1、在用户完成登录后,程序会把用户相关…...
excel将文件夹下面的表格文件指定名称的sheet批量导出到指定文件中,并按照文件名保存在新文件的不同sheet中
excel将文件夹下面的表格文件指定名称的sheet批量导出到指定文件中,并按照文件名保存在新文件的不同sheet中 import pandas as pd import ositems os.listdir("./") sheetname"" for item in items:if item.__contains__(xls):dfpd.read_exc…...
IIS管理器无法打开。启动后,在任务栏中有,但是窗口不见了
找到IIS管理器启动程序的所在位置 并在cmd命令行中调用 inetmgr.exe /reset 进行重启 先查看IIS管理器属性,找到其位置 管理员模式打开cmd命令行,并切换到上面的文件夹下运行Inetmgr.exe /reset 运行完成后可以重新看到IIS窗口 原因:由于某…...
使用VBA实现快速模糊查询数据
实例需求:基础数据保存在Database工作表中,如下图所示。 基础数据有37个字段,上图仅展示部分字段内容,下图中黄色字段为需要提取的数据字段。 在Search工作表B1单元格输入查询关键字Title和Genre字段中搜索关键字,包…...
spring boot flowable多人前加签
1、前加签插件 package com.xxx.flowable.cmd;import com.xxx.auth.security.user.SecurityUser; import com.xxx.commons.ApplicationContextHolder; import com.google.common.collect.Lists; import org.apache.commons.collections.CollectionUtils; import org.apache.co…...
结构体运算符重载
1.降序 struct Point{int x, y;//重载比较符bool operator < (const Point &a) const{return x > a.x;//当前元素大时,是降序} };2.升序 struct Point{int x, y;//重载比较符 // bool operator < (const Point &a) const{ // return x…...
幽默直观的文档作者注释
注释是程序中非常重要的一部分,在不同的编程语言中,注释的风格和语言描述会有所不同。以下是一些常用的注释风格和语言描述: 直观注释:这种注释使用简洁、明了的语言,帮助读者快速地理解代码。例如,代码中…...
前端开发网站推荐
每个人都会遇见那么一个人,永远无法忘却,也永远不能拥有。 以下是一些可以用来查找和比较前端框架的推荐网站: JavaScript框架比较: 这些网站提供了对不同JavaScript框架和库的详细比较和评估。 JavaScripting: 提供了大量的JavaS…...
Java - Mysql数据类型对应
Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...
Nginx server_name 配置说明
Nginx 是一个高性能的反向代理和负载均衡服务器,其核心配置之一是 server 块中的 server_name 指令。server_name 决定了 Nginx 如何根据客户端请求的 Host 头匹配对应的虚拟主机(Virtual Host)。 1. 简介 Nginx 使用 server_name 指令来确定…...
【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...
Matlab | matlab常用命令总结
常用命令 一、 基础操作与环境二、 矩阵与数组操作(核心)三、 绘图与可视化四、 编程与控制流五、 符号计算 (Symbolic Math Toolbox)六、 文件与数据 I/O七、 常用函数类别重要提示这是一份 MATLAB 常用命令和功能的总结,涵盖了基础操作、矩阵运算、绘图、编程和文件处理等…...
C++八股 —— 单例模式
文章目录 1. 基本概念2. 设计要点3. 实现方式4. 详解懒汉模式 1. 基本概念 线程安全(Thread Safety) 线程安全是指在多线程环境下,某个函数、类或代码片段能够被多个线程同时调用时,仍能保证数据的一致性和逻辑的正确性…...
基于matlab策略迭代和值迭代法的动态规划
经典的基于策略迭代和值迭代法的动态规划matlab代码,实现机器人的最优运输 Dynamic-Programming-master/Environment.pdf , 104724 Dynamic-Programming-master/README.md , 506 Dynamic-Programming-master/generalizedPolicyIteration.m , 1970 Dynamic-Programm…...
Hive 存储格式深度解析:从 TextFile 到 ORC,如何选对数据存储方案?
在大数据处理领域,Hive 作为 Hadoop 生态中重要的数据仓库工具,其存储格式的选择直接影响数据存储成本、查询效率和计算资源消耗。面对 TextFile、SequenceFile、Parquet、RCFile、ORC 等多种存储格式,很多开发者常常陷入选择困境。本文将从底…...
服务器--宝塔命令
一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行! sudo su - 1. CentOS 系统: yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...
技术栈RabbitMq的介绍和使用
目录 1. 什么是消息队列?2. 消息队列的优点3. RabbitMQ 消息队列概述4. RabbitMQ 安装5. Exchange 四种类型5.1 direct 精准匹配5.2 fanout 广播5.3 topic 正则匹配 6. RabbitMQ 队列模式6.1 简单队列模式6.2 工作队列模式6.3 发布/订阅模式6.4 路由模式6.5 主题模式…...
处理vxe-table 表尾数据是单独一个接口,表格tableData数据更新后,需要点击两下,表尾才是正确的
修改bug思路: 分别把 tabledata 和 表尾相关数据 console.log() 发现 更新数据先后顺序不对 settimeout延迟查询表格接口 ——测试可行 升级↑:async await 等接口返回后再开始下一个接口查询 ________________________________________________________…...



