基于静态顺序表实现通讯录

目录

一、设计框架

1、功能要求​

2、菜单函数的实现

二、头文件实现​

Contact.h 

SeqList.h

三、Test.h

四、通讯录的初始化和销毁

五、增加通讯录

六、在通讯录中查找姓名下标

七、删除通讯录

八、显示通讯录

九、查找通讯录


一、设计框架

test.c:通讯录的总体逻辑,测试通讯录的相关功能
contact.c:通讯录的实现模块
contact.h:通讯录的各种声明,包括库函数、自定义函数以及自定义结构体的声明

1、功能要求​

  1. 至少能够存储100个人的通讯信息​
  2. 能够保存用户信息:名字、性别、年龄、电话、地址等​
  3. 增加联系人信息​
  4. 删除指定联系人​
  5. 查找制定联系人​
  6. 修改指定联系人​
  7. 显示联系人信息​

2、菜单函数的实现

实现通讯录建立一个菜单是很重要的,并且菜单要包含通讯录所有的功能,以便于用户的操作

//通讯录菜单
void menu() {printf("*****************通讯录***************\n");printf("*******1.添加联系人  2.删除联系人*****\n");//ctrl+dprintf("*******3.修改联系人  4.查找联系人*****\n");//ctrl+dprintf("*******5.查看通讯录  0.  退 出  ******\n");//ctrl+dprintf("**************************************\n");
}

二、头文件实现​

问题:头文件包含嵌套

解决方案:前置声明

Contact.h 

#pragma once
//#include<stdio.h> //暂时加上#define NAME_MAX 100
#define GENDER_MAX 10
#define TEL_MAX 12
#define ADDR_MAX 100//通讯录数据类型
typedef struct PersonInfo
{char name[NAME_MAX];int age;char gender[GENDER_MAX];char tel[TEL_MAX];char addr[ADDR_MAX];
}Info;//使用顺序表的前置声明
struct SeqList;typedef struct SeqList Contact;//通讯里提供的操作//通讯录的初始化和销毁
void ContactInit(Contact* pcon);//实际初始化的还是顺序表
void ContactDesTroy(Contact* pcon);//增加、删除、修改、查找、查看通讯录
void ContactAdd(Contact* pcon);
void ContactDel(Contact* pcon);
void ContactModify(Contact* pcon);
void ContactFind(Contact* pcon);
void ContactShow(Contact* pcon);

SeqList.h

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include"Contact.h"//静态顺序表//#define N 100
//struct SeqList
//{
//	SLDataType a[N];
//	int size;
//};//动态顺序表//typedef int SLDataType;
typedef Info SLDataType;typedef struct SeqList
{SLDataType* arr; //存储数据的底层结构int capacity;    //记录顺序表的空间大小int size;        //记录顺序表当前有效的数据个数
}SL;//typedef struct SeqList SL;//初始化和销毁
void SLInit(SL* ps);
void SLDestroy(SL* ps);
void SLPrint(SL* ps); //保持接口一致性//顺序表的头部/尾部插入
void SLPushBack(SL* ps, SLDataType x);
void SLPushFront(SL* ps, SLDataType x);//顺序表的头部/尾部删除
void SLPopBack(SL* ps);
void SLPopFront(SL* ps);//指定位置之前插入数据
//删除指定位置数据
void SLInsert(SL* ps, int pos, SLDataType x);
void SLErase(SL* ps, int pos);
//int SLFind(SL* ps, SLDataType x);

三、Test.h

我们需要根据菜单里面的选项来选择进行我们需要实现的功能,比如我们想添加联系人,我们就输入1就会进行用户假如的操作,我们想退出程序我们输入0就可以退出。可以利用switch选择语句来实现各自的功能。

//#include"Contact.h"  //在SeqList.h文件中已经包了Contact.h
#include"SeqList.h"//通讯录菜单
void menu() {printf("*****************通讯录***************\n");printf("*******1.添加联系人  2.删除联系人*****\n");//ctrl+dprintf("*******3.修改联系人  4.查找联系人*****\n");//ctrl+dprintf("*******5.查看通讯录  0.  退 出  ******\n");//ctrl+dprintf("**************************************\n");
}
int main()
{int op = -1;//创建通讯录结构对象Contact con;ContactInit(&con);do {menu();printf("请选择您的操作:\n");scanf("%d", &op);switch (op){case 1://添加联系人ContactAdd(&con);break;case 2://删除联系人ContactDel(&con);break;case 3://修改联系人ContactModify(&con);break;case 4://查找联系人ContactFind(&con);break;case 5://查看通讯录ContactShow(&con);break;case 0://退出通讯录printf("通讯录退出中...\n");break;default:break;}} while (op != 0);//销毁通讯录ContactDesTroy(&con);return 0;
}

四、通讯录的初始化和销毁

  • ContactInit:初始化通讯录。它调用SLInit来确保通讯录(被视为顺序表)的内部状态是干净的,即没有分配内存且大小与容量均为0。
  • ContactDesTroy:销毁通讯录。它调用SLDestroy来释放通讯录占用的内存资源,并确保通讯录回到初始状态。
  • SLInit:初始化顺序表。它将顺序表的数组指针设为NULL,大小与容量均设为0,表示这是一个空表。
  • SLDestroy:销毁顺序表。它首先检查顺序表指针是否为空,然后释放数组内存,将数组指针设为NULL,并将大小与容量重置为0。
//通讯录的初始化和销毁
//SL* ps
void ContactInit(Contact* pcon) {SLInit(pcon);
}void ContactDesTroy(Contact* pcon) {SLDestroy(pcon);
}
//初始化和销毁
void SLInit(SL* ps) {ps->arr = NULL; //不是int 而是Info类型 ps->size = ps->capacity = 0;
}
void SLDestroy(SL* ps) {assert(ps);if (ps->arr) {free(ps->arr);}ps->arr = NULL;ps->size = ps->capacity = 0;
}

五、增加通讯录

这段代码主要实现了两个功能:添加联系人到通讯录(ContactAdd函数)和在顺序表尾部插入数据(SLPushBack函数)。

1. 添加联系人到通讯录(ContactAdd函数):

  •    首先,定义了一个`Info`类型的结构体变量`info`,用于存储输入的联系人信息。
  •    然后,通过一系列`printf`和`scanf`函数,从用户那里获取联系人的姓名、年龄、性别、电话和住址,并保存到`info`结构体中。
  •    最后,调用`SLPushBack`函数,将这个联系人的信息添加到通讯录(实际上是一个顺序表)中。

2. 在顺序表尾部插入数据(SLPushBack函数):

  •     首先,使用`assert`函数检查传入的顺序表指针`ps`是否为空。如果为空,程序将终止运行。这是一种防止程序出错的方式。
  •     然后,调用`SLCheckCapacity`函数检查顺序表的容量是否足够。如果不够,该函数可能会进行扩容操作。
  •     最后,如果顺序表的空间足够,就直接在尾部插入数据,并更新顺序表的大小。

//增加通讯录
void ContactAdd(Contact* pcon) {//创建联系人结构体变量Info info;printf("请输入联系人姓名:\n");scanf("%s", info.name);printf("请输入联系人年龄:\n");scanf("%d", &info.age);//其他数据都是数组,年龄不是printf("请输入联系人性别:\n");scanf("%s", info.gender);printf("请输入联系人电话:\n");scanf("%s", info.tel);printf("请输入联系人住址:\n");scanf("%s", info.addr);//保存数据到通讯录(顺序表)SLPushBack(pcon, info);
}
//顺序表的头部/尾部插入
void SLPushBack(SL* ps, SLDataType x) {//断言--粗暴的解决方式//assert(ps != NULL);assert(ps);//if判断--温柔的解决方式//if (ps == NULL) {//	return;//}//空间不够,扩容SLCheckCapacity(ps);//空间足够,直接插入ps->arr[ps->size++] = x;//ps->size++;
}

六、在通讯录中查找姓名下标

  • 使用 for 循环遍历通讯录中的每一个联系人。循环变量 i 从 0 开始,直到通讯录的大小 pcon->size。
  • 在每次循环中,使用 strcmp 函数比较当前联系人的姓名 pcon->arr[i].name 和要查找的姓名 name。strcmp 函数用于比较两个字符串是否相同。
  • 如果 strcmp 函数返回 0,表示找到了姓名匹配的联系人。此时,直接返回该联系人在通讯录中的下标 i。
  • 如果循环结束后仍然没有找到匹配的联系人,函数返回 -1,表示查找失败。
//在通讯录中查找姓名下标
int FindByName(Contact* pcon, char name[]) {for (int i = 0; i < pcon->size; i++){if (strcmp(pcon->arr[i].name, name) == 0) {//找到了,返回下标return i;}}return -1;
}

七、删除通讯录

  • 首先,函数会提示用户输入要删除的联系人的姓名。
  • 然后,它会调用 FindByName 函数(这个函数在之前的解释中已经介绍过,它的作用是在通讯录中查找指定姓名的联系人的下标)。
  • 如果 FindByName 函数返回的下标小于0,说明通讯录中没有这个姓名的联系人,函数就会打印一条提示信息,然后结束。
  • 如果找到了这个姓名的联系人(即 FindByName 返回的下标非负),函数就会调用 SLErase 函数来执行删除操作。
  • 删除操作成功后,函数会打印一条提示信息,告知用户联系人已经成功删除。
void ContactDel(Contact* pcon) {//删除之前一定要先查找//找到了,可以删除//找不到,不能执行删除printf("请输入要删除的联系人姓名:\n");char name[NAME_MAX];scanf("%s", name);int findIndex = FindByName(pcon, name);if (findIndex < 0) {printf("要删除的联系人不存在!\n");return;}//执行删除操作SLErase(pcon, findIndex);printf("联系人删除成功!\n");
}
  • 函数首先会检查传入的顺序表指针和位置参数是否有效。如果顺序表指针为空,或者位置参数超出了顺序表的大小范围,函数就会通过 assert 宏终止程序运行,防止发生错误。
  • 如果参数有效,函数就会从指定位置开始,将后面每个位置的数据都向前移动一个位置,覆盖掉当前位置的数据。这样就相当于删除了指定位置的数据。
  • 数据移动完成后,函数会将顺序表的大小减1,因为已经删除了一个数据。 
//删除指定位置数据
void SLErase(SL* ps, int pos) {assert(ps);assert(pos >= 0 && pos < ps->size);//pos以后的数据往前挪动一位for (int i = pos; i < ps->size - 1; i++){ps->arr[i] = ps->arr[i + 1];//ps->arr[i-2] = ps->arr[i-1];}ps->size--;
}

八、显示通讯录

void ContactShow(Contact* pcon) {//格式大家下去感兴趣去调整printf("%s %s %s %s %s\n", "姓名", "性别", "年龄", "电话", "住址");for (int i = 0; i < pcon->size; i++){printf("%s %s %d %s %s\n",pcon->arr[i].name,pcon->arr[i].gender,pcon->arr[i].age,pcon->arr[i].tel,pcon->arr[i].addr);}
}

九、查找通讯录

  • 调用 FindByName 函数(该函数在之前的代码段中定义),在通讯录中查找指定姓名的联系人的下标。将返回的下标存储在 findIndex 变量中。
  • 判断 findIndex 的值:
  • 如果 findIndex 小于 0,表示没有找到匹配的联系人,于是打印一条提示信息并结束函数。
  • 如果 findIndex 非负,表示找到了匹配的联系人,继续执行下一步。
  • 打印找到的联系人的信息:
  • 首先打印一列标题,包括“姓名”、“性别”、“年龄”、“电话”和“住址”。
  • 然后根据 findIndex 下标,从通讯录中获取对应联系人的信息,并打印出来。
void ContactFind(Contact* pcon) {char name[NAME_MAX];printf("请输入要查找的用户姓名:\n");scanf("%s", name);int findIndex = FindByName(pcon, name);if (findIndex < 0) {printf("该联系人不存在!\n");return;}//找到了,打印一下查找的联系人信息printf("%s %s %s %s %s\n", "姓名", "性别", "年龄", "电话", "住址");printf("%s %s %d %s %s\n",pcon->arr[findIndex].name,pcon->arr[findIndex].gender,pcon->arr[findIndex].age,pcon->arr[findIndex].tel,pcon->arr[findIndex].addr);
}

今天就先到这了!!!

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!

你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。

我的博客即将同步至腾讯云开发者社区,邀请大家一同入驻:

腾讯云自媒体分享计划 - 腾讯云开发者社区-腾讯云

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

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

相关文章

《尊思想人文地理环境》新书亮相,叶无为集30年智慧破解环境密码

在探索人与自然和谐共生的今天&#xff0c;叶无为教授的新作《尊思想人文地理环境》应时而生&#xff0c;为读者揭开了地理环境与人文发展之间深刻联系的神秘面纱。本书集结了作者三十多年的实战经验&#xff0c;通过易医体系对大自然的山川河流进行独到解析&#xff0c;融合传…

飞越天空之城

欢迎来到程序小院 飞越天空之城 玩法&#xff1a;左边的按钮是控制小人儿飞起来的方向的&#xff0c;右边的按钮是控制它飞的高度的&#xff0c; 左边控制在正上方时可以让小人儿沿着一个方向飞跃&#xff0c;否则会撞到两边的黑墙&#xff0c; 右边的按钮如果加足够的话&…

GPTs大受欢迎但问题多,企服厂商的AI Agent更被B端客户器重

2023年11月&#xff0c;OpenAI在首届开发者大会上推出了GPTs和Assitant API&#xff0c;不仅改写了AI Agent的构建范式&#xff0c;也把AI智能体的应用推向一个新高潮。 GPTs和GPT商店&#xff0c;使得用户无需编码通过自然语言就能创建并拥有多个专属私人助理&#xff0c;且可…

【Servlet】如何编写第一个Servlet程序

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【Servlet】 本专栏旨在分享学习Servlet的一点学习心得&#xff0c;欢迎大家在评论区交流讨论&#x1f48c; Servlet是Java编写的服务器端…

5.【SpringBoot3】文件上传

1. 文件上传到本地 需求分析 在用户更换头像或发布文章时&#xff0c;需要携带一个图片的 url 地址&#xff0c;该 url 地址是当用户访问文件上传接口&#xff0c;将图片上传成功后&#xff0c;服务器返回的地址。所以&#xff0c;后台需要提供一个文件上传接口&#xff0c;用…

Android HIDL概述与绑定模式的实现

一、前言 Android O(8.0) 版本之后&#xff0c;底层实现有了比较大的变化&#xff0c;最显著的一个方面就是 HIDL 机制的全面实施。本文对于理解系统源码中 Gnss、Usb、Camera 等模块的工作原理有极大帮助。 二、HIDL 设计目的 在 Android O(8.0) 之前系统的升级牵扯多方协作…

Python tkinter (5) 选项按钮与复选框

Python的标准Tk GUI工具包的接口 tkinter系列文章 python tkinter窗口简单实现 Python tkinter (1) —— Label标签 Python tkinter (2) —— Button标签 Python tkinter (3) —— Entry标签 Python tkinter (4) —— Text控件 目录 CheckButton 简单示例 获取选中 Ra…

RBD —— Visualizing fractured geometry

RBD Exploded View&#xff08;与Exploded View SOP类似&#xff09;从中心炸开几何体&#xff0c;以更好查看被破碎和约束的碎块&#xff1b; 可视化高精度和低精度几何体的不同&#xff0c;Show Proxy Geometry显示代理几何体&#xff1b; Show Constraints显示约束&#xff…

6.jmeter非GUI命令及Beanshell组件

一、非GUI&#xff08;界面&#xff09;命令详解 1. -n 使用非gui方式&#xff0c;不能单独使用&#xff0c;必须和-t&#xff08;指定jmeter的脚本&#xff09;一起用。 #cmd命令行模式下&#xff0c;进入存放测试jmx文件的目录下 jmeter -n -t hello.jmx只会生成一个log日…

数据结构实验八:排序的应用

目录 一、实验目的 二、实验原理 1.直接插入排序 2.快速排序 三、实验内容 实验1 代码 截图 实验2 代码 截图 一、实验目的 1、掌握排序的基本概念&#xff1b; 2&#xff0e;掌握并实现以下排序算法&#xff1a;直接插入排序、快速排序。 二、实验原理 1.直接插…

如何选择便捷安全的黄金交易平台?

黄金交易平台的介绍 黄金交易平台是一个提供方便、安全的方式进行黄金交易的网上平台。 投资者可以通过这些平台进行黄金的买卖&#xff0c;参与黄金市场的投资活动。 这些平台提供了一个简单易用的界面&#xff0c;让投资者可以方便地进行交易操作。 选择合适的黄金交易平台…

小土堆pytorch学习笔记002

1、TensorBoard的使用 &#xff08;1&#xff09;显示坐标&#xff1a; from torch.utils.tensorboard import SummaryWriter import numpy as np from PIL import Imagewriter SummaryWriter("logs") # 写入的位置 log_dir logs # writer.add_image() "…

【LeetCode: 148. 排序链表 + 链表 + 归并排序】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

今天来看看工商业储能收益模式有哪些

安科瑞武陈燕acrelcy 2023 年有望成为工商业储能的发展元年&#xff0c;主要原因2023年工商业储能的经济性有望大幅提升。工商业储能下游主要为工商业企业&#xff0c;投资是否具有经济性是工商业需求的核心因素之一&#xff0c;而2023年工商业储能经济性或将显著提升&#xf…

shared_ptr 与 unique_ptr 的转换 笔记

推荐B站文章&#xff1a; 6.shared_ptr与unique_ptr_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV18B4y187uL?p6&vd_sourcea934d7fc6f47698a29dac90a922ba5a3我的往期文章&#xff1a; 独占指针&#xff1a;unique_ptr 与 函数调用-CSDN博客https://blog.csdn.n…

【Docker】构建镜像

一般来说我们不需要自己构建镜像&#xff0c;这些镜像在docker hub上面已经有现成的了&#xff0c;比如常用的数据库、应用软件等。 某些情况我们还是需要自己来构建: 找不到现成的镜像&#xff0c;比如自己开发的应用程序 需要在镜像中加入特定的功能/软件 Docker提供了两种…

C++入门学习(十五)运算符

算术运算符&#xff1a;用于处理四则运算赋值运算符&#xff1a;用于将表达式的值赋给变量比较运算符&#xff1a;用于表达式的比较&#xff0c;并返回一个真值或假值逻辑运算符&#xff1a;用于根据表达式的值返回真值或假值 一、加减乘除 #include <iostream> #incl…

Ubuntu20.04 安装 ROS noetic + MAVROS

本文在 AlphaCatOvO【ROS】在 Ubuntu 20.04 安装 ROS 的详细教程 基础上&#xff0c;根据实际安装经验&#xff0c;稍微进行补充。 一、安装Ubuntu20.04 假设已经正确安装。 二、安装 ROS noetic 2.1 换源 执行 sudo apt update sudo mv /etc/apt/sources.list /etc/apt/…

A股市场风云!深圳开股票账户交易佣金费用最低是多少?

最近A股市场表现出色&#xff0c;呈现出风云状态&#xff01;投资者纷纷涌入股市&#xff0c;推动股指不断攀升。一方面&#xff0c;政府出台了一系列政策来稳定市场&#xff0c;如加强监管力度、推动资本市场改革等&#xff0c;为投资者提供了更多机会和便利条件。另一方面&am…

“情暖寒冬 温暖相伴”关爱慰问

岁末寒冬&#xff0c;春节将至&#xff0c;为持续巩固脱贫攻坚成果&#xff0c;扎实助力乡村振兴&#xff0c;市融媒体中心开展“情暖寒冬 温暖相伴”慰问活动。 1月25市融媒体中心联合我市志愿者协会在南村镇忽树庄村开展“情暖寒冬 温暖相伴”慰问活动&#xff0c;为脱贫户送…