运用动态内存实现通讯录(增删查改+排序)

目录

前言:

实现通讯录:

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!

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

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

相关文章

实现简单BS架构案例

BS架构简单通俗理解 就是 浏览器–服务器模式&#xff0c;浏览器 充当 我们的客户端。 目录 简单BS架构实现案例基本原理视图访问规则案例要求改造前服务端线程模版类 改造后(优化)优化策略服务端线程模版类 参考视频 简单BS架构实现案例 基本原理视图 注&#xff1a;服务器必…

【红外与可见光图像融合】离散平稳小波变换域中基于离散余弦变换和局部空间频率的红外与视觉图像融合方法(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

selenium下载安装 -- 使用谷歌驱动碰到的问题

安装教程参考: http://c.biancheng.net/python_spider/selenium.html 1. 谷歌浏览器和谷歌驱动版本要对应(但是最新版本谷歌对应的驱动是没有的,因此要下载谷歌历史其他版本): 谷歌浏览器历史版本下载: https://www.chromedownloads.net/chrome64win/谷歌浏览器驱动下载: http:…

基于web的医院预约挂号系统/医院管理系统

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;各行各业相继进入信息管理时代&a…

【GESP考级C++】1级样题 闰年统计

GSEP 1级样题 闰年统计 题目描述 小明刚刚学习了如何判断平年和闰年&#xff0c;他想知道两个年份之间&#xff08;包含起始年份和终止年份&#xff09;有几个闰年。你能帮帮他吗&#xff1f; 输入格式 输入一行&#xff0c;包含两个整数&#xff0c;分别表示起始年份和终止…

87、Redis 的 value 所支持的数据类型(String、List、Set、Zset、Hash)---->List相关命令

本次讲解要点&#xff1a; List相关命令&#xff1a;是指value中的数据类型 启动redis服务器&#xff1a; 打开小黑窗&#xff1a; C:\Users\JH>e: E:>cd E:\install\Redis6.0\Redis-x64-6.0.14\bin E:\install\Redis6.0\Redis-x64-6.0.14\bin>redis-server.exe redi…

MySQL约束

文章目录 简单介绍主键约束添加单列主键多列主键删除主键 自增长约束(auto_increment)语法&#xff1a;指定自增字段初始值 非空约束唯一约束(unique)默认约束(default)零填充约束(zerofill) 简单介绍 概念&#xff1a;表中数据的约束条件 作用&#xff1a;表在设计的时候加入…

[学习笔记]ARXML - Data Format

参考AUTOSAR文档&#xff1a; https://www.autosar.org/fileadmin/standards/R22-11/FO/AUTOSAR_TPS_ARXMLSerializationRules.pdfhttps://www.autosar.org/fileadmin/standards/R22-11/FO/AUTOSAR_TPS_ARXMLSerializationRules.pdf 编码 arxml只允许使用UTF-8编码&#xff…

【Java 进阶篇】JDBC ResultSet 遍历结果集详解

在Java数据库编程中&#xff0c;经常需要执行SQL查询并处理查询结果。ResultSet&#xff08;结果集&#xff09;是Java JDBC中用于表示查询结果的关键类之一。通过遍历ResultSet&#xff0c;我们可以访问和操作从数据库中检索的数据。本文将详细介绍如何使用JDBC来遍历ResultSe…

ubuntu 18.04 LTS安装opencv 3.4.16 + opencv_contrib 3.4.16

1.下载 opencv 3.4.16 opencv_contrib 3.4.16 其中&#xff0c;opencv_contrib解压后的多个文件夹复制到opencv内、合并 2.安装 参考博文&#xff1a; https://zhuanlan.zhihu.com/p/650792342 https://zhuanlan.zhihu.com/p/87197806 其中 &#xff08;1&#xff09;cmake前…

2023年中国金刚石工具产量、需求量、市场规模及行业细分产品规模[图]

金刚石工具就是以金刚石为增强体&#xff0c;以金属或聚合物为基体的一类功能复合材料。金刚石工具已经成为当今公认的、唯一有效的硬脆非金属材料加工工具&#xff0c;比如&#xff0c;只有用金刚石刀具能加工超硬的陶瓷&#xff0c;尚无其他代用品。金刚石工具根据其用途的不…

【VIM】初步认识VIM-2

2-6 Vim 如何搜索替换_哔哩哔哩_bilibili 1-6行将self改成this 精确替换quack单词为交

湖南特色农产品销售系统APP /基于android的农产品销售系统/基于android的购物系统

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的APP应运而生&#xff0c;各行各业相继进入信息管理时代&#x…

【QT】自定义组件ui类添加到主ui界面方法

1.添加自定义组件到项目中 add new选择如下 写好类方法&#xff0c;确定即可 2.将新创建的ui类加入到主ui界面 选中新创建ui类的父类空块&#xff0c;右键选择提升为 选择并添加新创建的类

如何搭建团队知识库?试试新的工具和方法吧!

知识本身没有价值&#xff0c;只有被利用的知识才能发挥作用。我们经常见到有许多“宏伟”的团队知识库&#xff0c;但是从来没有人去用…… 搭建团队知识库 没有人用的团队知识库存在的问题是“我们知道所有问题的答案&#xff0c;就是不知道问题是什么”。如何建立团队知识库…

NLP中token总结

Token 可以被理解为文本中的最小单位。在英文中&#xff0c;一个 token 可以是一个单词&#xff0c;也可以是一个标点符号。在中文中&#xff0c;通常以字或词作为 token。ChatGPT 将输入文本拆分成一个个 token&#xff0c;使模型能够对其进行处理和理解 在自然语言处理&#…

vertx的学习总结2

一、什么是verticle verticle是vertx的基本单元&#xff0c;其作用就是封装用于处理事件的技术功能单元 &#xff08;如果不能理解&#xff0c;到后面的实战就可以理解了&#xff09; 二、写一个verticle 1. 引入依赖&#xff08;这里用的是gradle&#xff0c;不会吧&#…

18scala笔记

Scala2.12 视频地址 1 入门 1.1 发展历史 … 1.2 Scala 和 Java Scala Java 编写代码使用scalac编译成.class字节码文件scala .class文件 执行代码 1.3 特点 1.4 安装 视频地址 注意配置好环境变量 简单代码 1.5 编译文件 编译scala文件会产生两个.class文件 使用java…

【初始Linux】上

初始Linux上 一、Linux背景1.1 UNIX发展的历史1.2 UNIX发展的历史 二、开源三、官网Linux官网 四、企业应用现状五、发行版本六、 os概念&#xff0c;定位 本博客简介 初始Linux操作系统初识shell命令 ,了解若干背景知识。使用常用Linux命令了解Linux权限概念与思想,能深度理解…

Java | Maven(知识点查询)

文章目录 Maven知识速查1. Maven概述2. Maven的作用3. Maven的下载4. Maven的环境配置5. Maven 的基础组成5.1 Maven仓库5.1.1 本地仓库配置&#xff1a;5.1.2 中央仓库配置&#xff1a;5.1.3 镜像仓库配置 5.2 Maven坐标 6. Maven项目6.1 手工创建Maven项目6.2 自动构建项目 7…