C语言学习记录—进阶作业(通讯录静态版本)

通讯录

实现一个通讯录
人的信息:名字+年龄+性别+电话+地址
1. 存放100个人的信息
2. 增加联系人
3. 删除指定联系人
4. 查找联系人
5. 修改联系人
6. 排序
7. 显示联系人

test.c 测试功能
contact.c 通讯录相关的实现
contact.h 通讯录相关的声明

整体设计思路:

1. 程序主函数类似三子棋。程序开始时展示菜单,提供增加联系人、删除指定联系人、 查找联系人、修改联系人、排序、显示联系人、退出功能。即使用do while循环保证程序可以重复执行,用switch语句实现选择功能。

2. 创建联系人信息的结构体。

3. 创建通讯录的结构体。即使用联系人结构体创建数组,并且创建一个计数器变量,用于记录通讯录中联系人的个数(即有效信息)。

4. 实现上述提到的各功能。

contact.h文件

#pragma once
#include <string.h>
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 10
#define MAX_TELE 12
#define MAX_ADDR 30//类型的声明
typedef struct PeoInfo//创建人的信息并重命名
{char name[MAX_NAME];int age;char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];
}PeoInfo;//通讯录
typedef struct Contact
{PeoInfo data[MAX];//存放100个人的信息int count;//记录当前通讯录中实际人的个数。//因为每次增删时,里面的有效信息个数在变化,所以需要一个变量来记录有个人的信息//又因为上面两个数据紧密相关,所以封装到一层
}Contact;//初始化通讯录
void InitContact(Contact* pc);//增加联系人到通讯录
void AddContact(Contact* pc);//打印通讯录
void ShowContact(const Contact* pc);//删除指定联系人
void DelContact(Contact* pc);//查找指定联系人
void SearchContact(Contact* pc);//修改指定联系人
void ModifyContact(Contact* pc);//按名字排序
void SortContact(Contact* pc);

contact.c文件

#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"void InitContact(Contact* pc)
{assert(pc);//下方要对结构体指针解引用,所以需要断言pc->count = 0;memset(pc->data, 0, sizeof(pc->data));//memset是逐个字节赋值//int类型的一般都是赋值0或 - 1,其他的值都不行。//因为 - 1和0转化成二进制后每一位都是一样的,设int型占4个字节,则 - 1 = 0XFFFFFFFF, 0 = 0X00000000。
}void AddContact(Contact* pc)
{assert(pc);//添加之前需要判断通讯录是否已满if (pc->count == MAX){printf("通讯录已满,无法添加\n");return;}//添加时需要考虑新的信息存放位置,而count恰好可以作为一个标识符。//比如count为0时,说明通讯录没有信息,此时可以添加在下标为0的位置printf("请输入名字:>");scanf("%s", pc->data[pc->count].name);//name是数组名,数组名就是首元素地址,所以不需要取地址printf("请输入年龄:>");scanf("%d", &(pc->data[pc->count].age));printf("请输入性别:>");scanf("%s", pc->data[pc->count].sex);printf("请输入电话:>");scanf("%s", pc->data[pc->count].tele);printf("请输入地址:>");scanf("%s", pc->data[pc->count].addr);//添加完信息后,需要修改count的值pc->count++;printf("增加成功\n");
}void ShowContact(const Contact* pc)
{printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");int i = 0;for (i = 0; i < pc->count; i++)//通讯录中的信息数量是count个{//打印data数组下标为i的信息printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\n",  pc->data[i].name, pc->data[i].age, pc->data[i].sex, pc->data[i].tele, pc->data[i].addr);}
}static int FindByName(Contact* pc, char name[])//不暴露此函数,所以不声明。此函数只是给下方删除函数使用。且使用static修饰,只能在此源文件下看到
{int i = 0;for (i = 0; i < pc->count; i++){if (0 == strcmp(pc->data[i].name, name)){return i;}}return -1;
}void DelContact(Contact* pc)
{assert(pc);char name[MAX_NAME] = { 0 };printf("请输入删除的姓名:>");scanf("%s", name);if (pc->count == 0){printf("通讯录为空,无法删除\n");return;}//删除//1.查找int pos = FindByName(pc, name);//去pc指向的data数组里查找名字if (pos == -1){printf("要删除的人不存在\n");return;}//2.删除//删除就是从指定位置(即要删除的元素位置)开始,要删除元素的后一个位置的信息往前移动覆盖,直到后面所有信息都往前移动一个位置int i = 0;//这里最后位置应该是最大下标-1,因为如果遍历到最后一个下标,那么下方i+1就会越界//并且如果删除最后一个元素,也不需要前移覆盖for (i = pos; i < pc->count - 1; i++){pc->data[i] = pc->data[i + 1];}pc->count--;printf("删除成功\n");
}void SearchContact(Contact* pc)
{assert(pc);char name[MAX_NAME] = { 0 };printf("请输入查找的姓名:>");scanf("%s", name);//1.查找int pos = FindByName(pc, name);//去pc指向的data数组里查找名字if (pos == -1){printf("要查找的人不存在\n");return;}//2.打印printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");//打印data数组下标为pos的信息printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\n", pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}void ModifyContact(Contact* pc)
{//1.查找assert(pc);char name[MAX_NAME] = { 0 };printf("请输入修改的姓名:>");scanf("%s", name);int pos = FindByName(pc, name);//去pc指向的data数组里查找名字if (pos == -1){printf("要修改的人不存在\n");return;}//2.修改printf("要修改人的信息已查找到,接下来开始修改\n");printf("请输入名字:>");scanf("%s", pc->data[pos].name);//name是数组名,数组名就是首元素地址,所以不需要取地址printf("请输入年龄:>");scanf("%d", &(pc->data[pos].age));printf("请输入性别:>");scanf("%s", pc->data[pos].sex);printf("请输入电话:>");scanf("%s", pc->data[pos].tele);printf("请输入地址:>");scanf("%s", pc->data[pos].addr);printf("修改成功\n");
}cmp_peo_by_name(const void* e1, const void* e2)
{return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}void SortContact(Contact* pc)
{assert(pc);qsort(pc->data, pc->count, sizeof(PeoInfo), cmp_peo_by_name);printf("排序成功\n");
}

test.c文件

#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"//实现一个通讯录
//人的信息:
//名字+年龄+性别+电话+地址
//1. 存放100个人的信息
//2. 增加联系人
//3. 删除指定联系人
//4. 查找联系人
//5. 修改联系人
//6. 排序
//7. 显示联系人//test.c 测试功能
//contact.c 通讯录相关的功能
//contact.h 通讯录相关的声明//这个是静态版本 - 通讯录大小无法改变,固定只能存100个人的信息
//动态版本 - 可以根据实际情况决定,不够可以继续开辟空间
//文件版本 - 程序退出依然可以保存信息enum Option
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};//菜单
void menu()
{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");
}int main()
{int input = 0;Contact con;//通讯录//初始化通讯录InitContact(&con);//不分开初始化的原因:尽量模块化工作,函数初始化更方便do{menu();printf("请输入:>");scanf("%d", &input);switch (input){//case 1://这里的常量1 2 3 4不能直观看出来对应菜单的哪个功能,所以可以使用枚举case ADD:AddContact(&con);//增加需要修改通讯录,所以传址调用break;case DEL:DelContact(&con);break;case SEARCH:SearchContact(&con);break;case MODIFY:ModifyContact(&con);break;case SHOW:ShowContact(&con);//虽然不修改,但是传地址提高效率break;case SORT:SortContact(&con);break;case EXIT:printf("退出通讯录\n");break;default:printf("选择错误\n");break;}} while (input);return 0;
}

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

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

相关文章

【LeetCode】203. 移除链表元素(简单)——代码随想录算法训练营Day03

题目链接&#xff1a;203. 移除链表元素 题目描述 给你一个链表的头节点 head 和一个整数 val &#xff0c;请你删除链表中所有满足 Node.val val 的节点&#xff0c;并返回 新的头节点 。 示例 1&#xff1a; 输入&#xff1a;head [1,2,6,3,4,5,6], val 6 输出&#xff…

Android 12.0 系统开启和关闭黑白模式主题功能

1.概述 在12.0的rom系统开发定制化中,在系统SystemUI的下拉状态栏中,产品开发功能需求要求添加黑白模式功能开关的功能,就是打开黑白模式,系统颜色就会变成黑白颜色, 关闭黑白模式开关系统就会变成彩色模式,所以就需要了解下系统是怎么设置黑白模式和彩色模式的,然后添…

什么是国密算法

国密算法是指由中国国家密码管理局发布的密码算法标准&#xff0c;旨在保障国家信息安全。目前&#xff0c;国家密码管理局已发布了一系列国产商用密码标准算法&#xff0c;包括SM1&#xff08;SCB2&#xff09;、SM2、SM3、SM4、SM7、SM9以及祖冲之密码算法&#xff08;ZUC)等…

SGL-110型定时限过流继电器 额定电流5A 额定电压220V 交直流通用 板前接线

系列型号 LGY-110零序过电压继电器&#xff1b; LGL-110零序过电压继电器&#xff1b; LGL-110/AC零序过电压继电器&#xff1b; LGL-110零序过电流继电器 板前接线 &#xff11; 应用 LGL-110 型零序过电流继电器用作线路和电力设备的零序过电流保护。 LGY-110 型零序过…

SQL Server的彻底卸载的方式

这篇文章主要介绍了SQL Server的彻底卸载的方式&#xff0c;具有很好的参考价值&#xff0c;希望对大家有所帮助。如有错误或未考虑完全的地方&#xff0c;望不吝赐教 SQL Server的彻底卸载与再次安装 可能大家已经有深刻体会&#xff0c;SQL Server的卸载十分繁琐。最让人头…

SQL执行时间过长如何优化

这个问题&#xff0c;其实跟慢 SQl 排查解决有点像。可以从以下这几个方面入手&#xff1a; 确定瓶颈 首先查看 MySQL 日志、慢查询日志、explain 分析 SQL 的执行计划、profile 分析执行耗时、Optimizer Trace分析详情等操作&#xff0c;确定查询执行的瓶颈在哪里。只有确定…

【AWS】使用亚马逊云服务器创建EC2实例

目录 前言为什么选择 Amazon EC2 云服务器搭建 Amazon EC2 云服务器注册亚马逊账号登录控制台服务器配置免费套餐预览使用 Amazon EC2 云服务器打开服务器管理界面设置服务器区域填写实例名称选择服务器系统镜像选择实例类型创建密钥对网络设置配置存储启动实例查看实例 总结 前…

第22集《佛法修学概要》

请大家打开讲义第六十一页。 我们这一科讲到归敬三宝。前面讲到&#xff0c;我们在心中能够受持“常住”两个字&#xff0c;就能够远离三恶道。“常住”是针对生灭来说的&#xff0c;我们的心是没有常住的。凡夫的心深受感受的刺激&#xff0c;一接触外境就带动感受&#xff0…

Eclipse插件UCdetector清理无用JAVA代码

下载插件 UCDetector - Browse /ucdetector at SourceForge.net 目前最新版本是2017年的2.0.0 保存 Eclipse/dropins 重启 操作 在项目上右键

ESU毅速丨复杂结构模具可尝试3D打印随形水路

冷却水路对模具的生产效率影响巨大&#xff0c;一些结构复杂、骨位深的模具常规水路加工困难且冷却效果不理想&#xff0c;这时可尝试3D打印来制造水路。3D打印技术可以制造出具有复杂内部结构和任意几何形状的部件&#xff0c;特别适合结构复杂、骨位深、薄壁等特征的模具水路…

Python字符串的编码和解码

不同计算机之间进行数据传输&#xff0c;实际上传输的是二进制数据。 一.字符串的编码 将str类型转换成bytes类型&#xff0c;需要用到字符串的encode&#xff08;&#xff09;方法 语法格式&#xff1a; Str.encode(encoding’utf-8’, Errors’strict/ignore/replace’) …

回归预测 | Matlab基于SO-GRU蛇群算法优化门控循环单元的数据多输入单输出回归预测

回归预测 | Matlab基于SO-GRU蛇群算法优化门控循环单元的数据多输入单输出回归预测 目录 回归预测 | Matlab基于SO-GRU蛇群算法优化门控循环单元的数据多输入单输出回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab基于SO-GRU蛇群算法优化门控循环单元的数…

jdk、tomcat、mysql的安装windows项目部署

文章目录 1、安装jdk2、tomcat安装3、MySQL安装3、外部访问数据库 1、安装jdk 1.双击运行jdk-8u144进行一个安装 2.一直点击下一步&#xff0c;到修改路径那个地方把他的存放路径改到D盘 3.找到我们刚刚修改的那个路径点进bin目录然后复制该路径进行一个环境变量配置4.找到我的…

golang学习笔记——go语言多文件项目运行的四种方式

go语言多文件运行技巧 有两个源码文件的go语言项目如何运行? go.modmain.go Trie.go 如何直接运行go run main.go会提示找不到文件。 # 在windows10下运行 $ go run main.go # command-line-arguments .\main.go:6:9: undefined: Constructor是真的找不到文件吗。其实不是。…

锁定查询功能,完成查询后防止他人查询

查询者想要实现自己查询完成后&#xff0c;任何人都无法再次查询&#xff0c;发布者应该如何设置&#xff1f;易查分的【锁定查询功能】就可实现&#xff0c;本次就来介绍如何使用此功能。 &#x1f4cc;使用教程 &#x1f512;锁定查询功能介绍 ✅用户自主锁定&#xff1a;开启…

如何基于 Gin 封装出属于自己 Web 框架?

思路 在基于 Gin 封装出属于自己的 Web 框架前&#xff0c;你需要先了解 Gin 的基本用法和设计理念。 然后&#xff0c;你可以通过以下步骤来封装自己的 Web 框架&#xff1a; 封装路由&#xff1a;Gin 的路由是通过 HTTP 方法和 URL 路径进行匹配的&#xff0c;你可以根据自己…

观测云产品更新 | 日志、场景仪表板、监控器等

观测云更新 用户访问监测 &#xff08;RUM &#xff09; 公网 Dataway 支持 ip 转换成地理位置信息。 日志 > 查看器详情页 1、新增 BPF 网络日志采集及日志详情页&#xff0c;支持 Json 格式转化&#xff1b; 2、上述 1 中的日志详情页中新增可读的展示模式&#xff0c…

【驱动】TI AM437x(内核调试-05):创建设备节点到/proc进程文件系统中

1、接口 1.1 proc_create 定义在:include/linux/proc_fs.h /********************************************************************** * @ name:节点名 * @ mode:权限位 * @ parent: 父目录 * @ proc_fops:文件操作结构体 ****************************************…

2-认识小程序项目

基本结构 myapp├─miniprogram┊ └──pages┊ ┊ └──index┊ ┊ ┊ ├──index.json┊ ┊ ┊ ├──index.ts┊ ┊ ┊ ├──index.wxml┊ ┊ ┊ └──index.wxss┊ ┊ └──logs┊ ┊ ├──index.json┊ ┊ ├──index.ts┊ ┊ ├…

MQ消息堆积如何解决?

RabbitMQ如果有100万消息堆积在MQ&#xff0c;如何解决&#xff1f; 当生产者发送消息的速度超过消费者处理消息的速度&#xff0c;就会导致队列中的消息堆积&#xff0c;只到队列存储消息达到上限。之后发送的消息就会成为死信&#xff0c;可能会被丢弃&#xff0c;这就是消息…