运用动态内存实现通讯录(增删查改+排序)
目录
前言:
实现通讯录:
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…...
云计算——弹性云计算器(ECS)
弹性云服务器:ECS 概述 云计算重构了ICT系统,云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台,包含如下主要概念。 ECS(Elastic Cloud Server):即弹性云服务器,是云计算…...

对WWDC 2025 Keynote 内容的预测
借助我们以往对苹果公司发展路径的深入研究经验,以及大语言模型的分析能力,我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际,我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测,聊作存档。等到明…...

srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...
关于 WASM:1. WASM 基础原理
一、WASM 简介 1.1 WebAssembly 是什么? WebAssembly(WASM) 是一种能在现代浏览器中高效运行的二进制指令格式,它不是传统的编程语言,而是一种 低级字节码格式,可由高级语言(如 C、C、Rust&am…...

深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南
🚀 C extern 关键字深度解析:跨文件编程的终极指南 📅 更新时间:2025年6月5日 🏷️ 标签:C | extern关键字 | 多文件编程 | 链接与声明 | 现代C 文章目录 前言🔥一、extern 是什么?&…...

10-Oracle 23 ai Vector Search 概述和参数
一、Oracle AI Vector Search 概述 企业和个人都在尝试各种AI,使用客户端或是内部自己搭建集成大模型的终端,加速与大型语言模型(LLM)的结合,同时使用检索增强生成(Retrieval Augmented Generation &#…...

Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习)
Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习) 一、Aspose.PDF 简介二、说明(⚠️仅供学习与研究使用)三、技术流程总览四、准备工作1. 下载 Jar 包2. Maven 项目依赖配置 五、字节码修改实现代码&#…...
JavaScript基础-API 和 Web API
在学习JavaScript的过程中,理解API(应用程序接口)和Web API的概念及其应用是非常重要的。这些工具极大地扩展了JavaScript的功能,使得开发者能够创建出功能丰富、交互性强的Web应用程序。本文将深入探讨JavaScript中的API与Web AP…...
08. C#入门系列【类的基本概念】:开启编程世界的奇妙冒险
C#入门系列【类的基本概念】:开启编程世界的奇妙冒险 嘿,各位编程小白探险家!欢迎来到 C# 的奇幻大陆!今天咱们要深入探索这片大陆上至关重要的 “建筑”—— 类!别害怕,跟着我,保准让你轻松搞…...

什么是VR全景技术
VR全景技术,全称为虚拟现实全景技术,是通过计算机图像模拟生成三维空间中的虚拟世界,使用户能够在该虚拟世界中进行全方位、无死角的观察和交互的技术。VR全景技术模拟人在真实空间中的视觉体验,结合图文、3D、音视频等多媒体元素…...