C语言 通讯录管理 完整代码

这份代码,是我从网上找的。目前是能运行。我正在读。有些不懂的地方,等下再记录下来。
有些地方的命名,还需要重新写一下。

比如:

PersonInfo* info = &address_book->all_address[address_book->size];

应该改为:

PersonInfo* info = &(address_book->all_address[address_book->size]);

就是加上一个括号。更清晰一些。

效果图:

#include<stdio.h>
#include<string.h>#define Max_SIZE 50 //定义通讯录的大小typedef struct PersonInfo {char name[100];char phone[100];
}PersonInfo;typedef struct AddressBook {PersonInfo all_address[Max_SIZE];int size;
}AddressBook;
//size 的含义是:
//数组 all_address 下标范围在 [0,size) 内的元素是有意义的
//                            [size, 200) 是我们没有用到的//初始化
void init(AddressBook* address_book) {address_book->size = 0;//尽量少用 magic number(不明含义的数字)for (int i = 0; i < Max_SIZE; i++) {strcpy(address_book->all_address->name, " ");strcpy(address_book->all_address->phone, " ");}}int Menu(void) {printf("======================\n");printf("***     0.退出     ***\n");printf("***  1.新增联系人  ***\n");printf("***  2.删除联系人  ***\n");printf("***  3.查找联系人  ***\n");printf("***  4.修改联系人  ***\n");printf("***  5.打印联系人  ***\n");printf("***  6.清除联系人  ***\n");printf("***  7.排序联系人  ***\n");printf("======================\n");printf("请输入你的选择:");int choice;scanf("%d", &choice);return choice;
}void AddPersonInfo(AddressBook* address_book) {printf("新增联系人\n");if (address_book->size >= Max_SIZE) {printf("通讯录已满,请先清除!\n");return;//虽然函数类型是 void 但是也是可以用 return 滴}PersonInfo* info = &address_book->all_address[address_book->size];printf("请输入联系人姓名:");scanf("%s", info->name);printf("请输入联系人电话:");scanf("%s", info->phone);address_book->size++;
}void DelPersonInfo(AddressBook* address_book) {//删除的方法很多,可以根据姓名,电话,序号等等来删除,//这里我们就用我个人比较常用的 搜索名字的删除方法char search_name[100] = { 0 };printf("删除联系人\n");printf("请输入联系人姓名:");scanf("%s", search_name);int name_exist = FindName(address_book, search_name);//这里注意 FindName 返回值设定,要被删除的元素下标可能是 0,这种情况下会返回 0 if (name_exist == -1) {printf("该联系人不存在!\n");return;}//删除了相同姓名的第一个后,继续寻找改名字,如果找到了,重复上面的操作,如果没找到,退出循环//相同的姓名的情况比较复杂,在修改,查找,排序等等场景都会带来麻烦,//而且平时你的通讯录中难道会将两个相同姓名的人的备注写成一样的吗?//所以,我仅仅在删除功能中实现一种针对相同姓名的情况的设计思路,后面的其他功能默认没有重复姓名的情况。while (name_exist != -1) {//将 all_address 数组的最后一个元素赋值给要删除的元素,完成删除//结构体类型是可以直接赋值的address_book->all_address[name_exist] = address_book->all_address[address_book->size - 1];address_book->size--;name_exist = FindName(address_book, search_name);}printf("删除成功!\n");}int FindName(AddressBook* address_book, char search_name[100]) {for (int i = 0; i < address_book->size; i++) {//找到返回数组下标if (strcmp(address_book->all_address->name, search_name) == 0) {return i;}}//没有找到,返回 -1return -1;
}void FindPersonInfo(AddressBook* address_book) {char search_name[100] = { 0 };//搜索人的方式也很多,我们这里用搜索名字的方法printf("更新联系人\n");printf("请输入人名:");scanf("%s", search_name);for (int i = 0; i < address_book->size; i++) {PersonInfo* info = &address_book->all_address[i];// 创建一个 PersonInfo 类型的变量简化程序,不然下面的姓名访问就太长了if (strcmp(info->name, search_name) == 0) {printf("[%d] %s  %s\n", i, info->name, info->phone);}}}void ModifyPersonInfo(AddressBook* address_book) {char search_name[100] = { 0 };int isjump = 1;printf("删除联系人\n");printf("请输入联系人姓名:");scanf("%s", search_name);int name_exist = FindName(address_book, search_name);if (name_exist == -1) {printf("该联系人不存在!\n");return;}//优化以下用户的体验printf("请输入新的姓名,输入 0 跳过:");scanf("%d", &isjump);if (isjump) {scanf("%s", address_book->all_address[name_exist].name);}printf("请输入新的电话,输入 0 跳过:");scanf("%d", &isjump);if (isjump) {scanf("%s", address_book->all_address[name_exist].phone);}printf("更新成功!\n");}void PrintPersonInfo(AddressBook* address_book) {PersonInfo* info;if (address_book->size == 0) {printf("当前没有联系人!\n");return;}printf("所有联系人信息如下:\n");for (int i = 0; i < address_book->size; i++) {info = &address_book->all_address[i];printf("[%2d]%4s  %s\n", i, info->name, info->phone);}
}void ClearPersonInfo(AddressBook* address_book) {//清除所有信息是一种 危险的行为,我们最好让用户确认一次//相比你应该在自己的手机上回复过出厂设置,系统应该会让你确认不止一次!int is_continue = 0;printf("清除所有联系人,你确定吗?输入 0 继续: ");scanf("%d", &is_continue);if(is_continue == 0) {//将 size 置为 0 即可,不过你也可以将数组的每个元素都进行重置address_book->size = 0;}printf("清除完成!\n");
}void SortPersonInfo(AddressBook* address_book) {printf("排序通讯录\n");for (int i = 0; i < address_book->size - 1; i++) {for (int j = 0; j < address_book->size - 1 - i; j++) {PersonInfo* info = &address_book->all_address[j];PersonInfo* info_next = &address_book->all_address[j + 1];//简单的用 strcmp 进行排序,不过排序的行为感觉是“未定义”的,//看着有规律,但再多试试会发现很多情况并没有规律。//也还有很多可以排血的函数。比如:strcoll,wcsscoll,wcscmp 这些只要你能弄懂,我想你也可以用。//或者有其他更好的实现办法,但这并不是我们在这里的重点。if (strcmp(info->name, info_next) > 0) {PersonInfo tmp;tmp = *info;*info = *info_next;*info_next = tmp;}}}printf("排序成功!\n");
}int main(int argc, char* argv[]) {AddressBook address_book;//声明一个函数指针类型typedef void (*Func)(AddressBook*);Func func_table[] = {NULL,AddPersonInfo,DelPersonInfo,FindPersonInfo,ModifyPersonInfo,PrintPersonInfo,ClearPersonInfo,SortPersonInfo,};//或者你也可以这么做://声明一个函数类型://typedef void (Func)(AddressBook*);//我们用的是指针数组,数组类型必须是指针类型,所以应该加上 * //Func* func_table[] = {//NULL,//AddPersonInfo,//DelPersonInfo,//FindPersonInfo,//ModifyPersonInfo,//PrintPersonInfo,//ClearPersonInfo,//SortPersonInfo,//}; init(&address_book);while (1) {int choice = Menu();if (choice < 0 || choice > 7) {printf("输入错误!\n");continue;}if (choice == 0) {printf("再见!\n");break;}func_table[choice](&address_book);}return 0;
}

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

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

相关文章

使用SpringCloud搭建分布式配置中心

在现代的分布式系统中&#xff0c;配置管理是一个非常重要的组成部分。传统的做法是将配置文件放在每个服务的本地进行配置&#xff0c;这样的做法在规模较小的系统中还能够接受&#xff0c;但是当系统规模逐渐扩大时&#xff0c;配置管理将变得非常困难&#xff0c;容易出错。…

QT--文件操作和文件读写

文件操作和文件读写 QFile 类用于对文件进行操作&#xff0c;它继承自 QIODevice&#xff0c;可以进行读写操作。主要用于打开、关闭、读取、写入和管理文件。 1. 首先要指定文件路径 QFile fn(“文件路径”);也可以通过文件对话框来选择文件getOpenFileName 函数原型 QStr…

【大数据面试题】37 Doris 是怎么保证性能的?

一步一个脚印&#xff0c;一天一道大数据面试题 博主希望能够得到大家的点赞收藏支持&#xff01;非常感谢 点赞&#xff0c;收藏是情分&#xff0c;不点是本分。祝你身体健康&#xff0c;事事顺心&#xff01; Doris 是当下大热的 MPP 数据库&#xff0c;下面来聊聊它如何保证…

AGI 之 【Hugging Face】 的【零样本和少样本学习】之一 [构建标记任务] / [ 基线模型 ] 的简单整理

AGI 之 【Hugging Face】 的【零样本和少样本学习】之一 [构建标记任务] / [ 基线模型 ] 的简单整理 目录 AGI 之 【Hugging Face】 的【零样本和少样本学习】之一 [构建标记任务] / [ 基线模型 ] 的简单整理 一、简单介绍 二、零样本学习 (Zero-shot Learning) 和少样本学习…

【博主推荐】HTML5实现简洁的实用的个人网站、个人主页七个页面源码

文章目录 1.设计来源1.1 个人主页界面1.2 关于我界面1.3 我的技能界面1.4 我的经验界面1.5 我的教育界面1.6 我的项目界面1.7 联系我界面 2.效果和源码2.1 动态效果2.2 源代码 源码下载万套模板&#xff0c;程序开发&#xff0c;在线开发&#xff0c;在线沟通 作者&#xff1a;…

ipsec协议簇(详解)

IPSEC协议簇 IPSEC协议簇 --- 基于网络层的&#xff0c;应用密码学的安全通信协议组 IPV6中&#xff0c;IPSEC是要求强制使用的&#xff0c;但是&#xff0c;IPV4中作为可选项使用 IPSEC可以提供的安全服务 机密性 --- 数据加密 完整性 --- 防篡改可用性 数据源鉴别 -- 身份…

长效代理IP如何选用及代理服务分析

在这个数据为王、信息瞬息万变的时代&#xff0c;长效代理IP成为了众多开发者、数据科学家乃至普通网民手中的一把利器。它不仅能帮助我们解决地域管理&#xff0c;还能在保护隐私的同时&#xff0c;确保数据传输的稳定与安全。但面对市面上琳琅满目的代理服务&#xff0c;如何…

IVI(In-Vehicle Infotainment,智能座舱的信息娱乐系统)

IVI能够实现包括三维导航、实时路况、辅助驾驶等在线娱乐功能。 IVI人机交互形式&#xff08;三板斧&#xff09;&#xff1a;声音、图像、文字 IVI人机交互媒介I&#xff08;四件套&#xff09;&#xff1a;中控屏幕&#xff08;显示、触控&#xff09;、仪表显示、语言、方…

目标检测 | YOLO v1、YOLO v2、YOLO v3与YOLO v3 SPP理论讲解

☀️教程&#xff1a;霹雳吧啦Wz ☀️链接&#xff1a;https://www.bilibili.com/video/BV1yi4y1g7ro?p1&vd_sourcec7e390079ff3e10b79e23fb333bea49d 一、YOLO v1 针对于two-stage目标检测算法普遍存在的运算速度慢的缺点&#xff0c;YOLO创造性的提出了one-stage目标检测…

2024-07-20 Unity插件 Odin Serializer2 —— 序列化教程

文章目录 1 由根对象决定序列化2 实现 Odin 序列化器2.1 继承已有序列化类2.2 自定义序列化类 3 避免 Unity 无限深度警告4 指定序列化秘钥4.1 External String Reference Resolver4.2 External GUID Reference Resolver4.3 External Index Reference Resolver 4 功能与限制4.1…

为什么我不建议用Excel做进销存系统?

进销存管理系统是一个企业中非常关键的部分&#xff0c;它涉及商品的采购、销售和库存管理等复杂流程。虽然EXCEL作为一个办公软件&#xff0c;它的通用性和灵活性使其能够处理这类数据&#xff0c;但实际上&#xff0c;使用它来构建专业的进销存管理系统存在一些明显的局限性。…

haproxy服务介绍

haproxy 搭建使用开启HAProxy的界面UI配置负载均衡配置web代理 HAProxy&#xff08;High Availability Proxy&#xff09;是一个高性能的TCP/HTTP负载均衡器和代理服务器&#xff0c;广泛用于提升Web应用的可用性和性能。[官网说明](https://docs.haproxy.org/2.8/intro.html#3…

NLP: 词袋模型和TFIDF模型

文章目录 词袋模型TF-IDF模型词汇表模型 词袋模型 文本特征提取有两个非常重要的模型&#xff1a; 词集模型&#xff1a;单词构成的集合&#xff0c;集合自然每个元素都只有一个&#xff0c;也即词集中的每个单词都只有一个。 词袋模型&#xff1a;在词集的基础上如果一个单词…

autoxjs的安装与配置

AutoxJs 是一个基于 JavaScript 的自动化工具&#xff0c;用于在 Android 平台上创建自动化脚本。它是在原 Auto.js 项目的基础上继续维护和升级而来的。 AutoxJs 的优势主要包括以下几点&#xff1a; 无需 root 权限&#xff1a;可以在没有 root 权限的设备上运行大部分功能&…

JavaWeb系列二十三: web 应用常用功能(文件上传下载)

文件上传下载 基本介绍文件上传基本原理文件上传应用实例文件上传注意事项和细节 文件下载基本原理文件下载应用实例文件下载注意事项 ⬅️ 上一篇: JavaWeb系列二十二: 线程数据共享和安全(ThreadLocal) &#x1f389; 欢迎来到 JavaWeb系列二十三: web 应用常用功能(文件上传…

创建最佳实践创建 XML 站点地图--SEO

您是否正在努力让您的网站被搜索引擎索引&#xff1f;您想提高您网站的搜索引擎知名度吗&#xff1f;如果是&#xff0c;您可能会错过 XML 站点地图的重要性。XML 站点地图在改善您网站的 SEO 方面发挥着至关重要的作用。‍ XML 站点地图是您网站结构的蓝图&#xff0c;可帮助…

YOLOv5项目梳理

1 项目介绍 参考项目&#xff1a;YOLO项目 1.1训练模型 YOLOv5模型 train.py 训练预训练模型 ... ... def parse_opt(knownFalse):# 命令行参数解析器初始化parser argparse.ArgumentParser()# 初始权重路径&#xff0c;默认为 ROOT / yolov5s.pt&#xff0c;用于指定模…

Navicat 17 for Mac 数据库管理软件

Mac分享吧 文章目录 效果一、准备工作二、开始安装1. 双击运行软件&#xff0c;将其从左侧拖入右侧文件夹中&#xff0c;等待安装完毕。2. 应用程序/启动台显示Navicat图标&#xff0c;表示安装成功。 二、运行测试运行后提示&#xff1a;“Navicat Premium.pp”已损坏&#x…

在qt的c++程序嵌入一个qml窗口

//拖拽一个QQuickWidget c端和qml通信的桥梁 找到qml的main.qml的路径 ui->quickWidget->setSource(QUrl::fromLocalFile("../../../code/main.qml"));// QML 与 Qt Widgets 通信//窗口就成了一个类实例对象pRoot (QObject*)ui->quickWidget->rootObje…

redis面试基础知识

redis的数据类型 Redis是一个key-value的数据库&#xff0c;key一般是String类型&#xff0c;不过value的类型多种多样&#xff1a; 五种基本数据类型 Redis 通用命令 通用指令是部分数据类型的&#xff0c;都可以使用的指令&#xff0c;常见的有&#xff1a; KEYS&#xff…