“富婆”通讯录——让你少奋斗50年

文章目录

  • 一、项目需求分析
  • 二、通讯录各功能实现思路及代码
    • 准备工作
    • 2.1、打印一个菜单,提供用户选择功能
    • 2.2、添加联系人信息
    • 2.3、删除联系人信息
    • 2.4、查询联系人信息
    • 2.5、修改联系人信息
    • 2.6、显示所有联系人信息
    • 2.7、对所有联系人信息进行排序整理
    • 2.8、删除所有联系人信息
  • 三、通讯录代码的整合


一、项目需求分析

在现在这个万物互联的时代,足不出户就能认识到全国各地的朋友,为了日后好联系,一般都会互留联系方式,这就需要使用通讯录了。通讯录对于我们每个人都是非常熟悉的了。最常见的就是手机中的电话薄。现在我们就根据手机中的电话薄的功能来模拟实现一个能让你少奋斗50年的富婆 通讯录。我们先看实现的通讯录都有哪些具体细节:

  1. 打印一个菜单,提供用户选择功能;
  2. 添加联系人信息;
  3. 删除联系人信息;
  4. 查询联系人信息;
  5. 修改联系人信息;
  6. 显示所有联系人信息;
  7. 对所有联系人信息进行排序整理;
  8. 删除所有联系人信息;
  9. 操作完毕可选择退出。

二、通讯录各功能实现思路及代码

准备工作

1. 定义数据类型

我们都在知道,一个人的通讯录信息有多种数据类型,因此先要定义一个结构体PeoInfo表示一个人的信息,在定义一个结构体Contact用来记录通讯录的数据、大小和容量。

#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
#define DEFAULT_SZ 3
#define INC_SZ 2//表示一个人的信息
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; //data指向了存放数据的空间int sz;        //记录通讯录中有效信息个数int capacity;  //通讯录当前的容量}Contact, * pContact;

2. 初始化通讯录

用malloc函数来动态管理内存空间,来满足你日益增长的富婆人数 ,来规划内存空间。
注意malloc函数返回值是指针,扩容失败会返回NULL指针,需要进行判断一下.

//初始化通讯录
//动态版本
//1.默认能够存放3个人的信息
//2.不够的话,每次增加2个人信息
void InitContact(Contact* pc)
{pc->data = (PeoInfo*)malloc(DEFAULT_SZ * sizeof(PeoInfo));if (pc->data == NULL){printf("通讯录初始化失败:%s\n",strerror(errno));return;}pc->sz = 0;pc->capacity = DEFAULT_SZ;}

2.1、打印一个菜单,提供用户选择功能

通讯录系统服务对象肯定不只是我们自己,别人是不清楚系统的功能的。为了让用户能够了解、使用通讯录的各项功能,需要程序一开始运行,就先打印一个菜单提示,指引使用者。
我们可以根据自己的喜好,用一些字符去画出一个菜单。

代码分析:

void menu()
{printf("=================================\n");printf("======       通讯录        ======\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");}

运行截图:
在这里插入图片描述

2.2、添加联系人信息

通讯录是要不断添加富婆信息进来的,首先判断通讯录存放信息是否达到上限,空间满了就调用CheckCapacity函数进行扩容,然后调用AddContact函数用于写入添加人的信息。
添加完后记得sz+1,即通讯录人数+1.

代码分析:

//扩容功能
int CheckCapacity(Contact* pc)
{if (pc->sz == pc->capacity){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeoInfo));if (ptr == NULL){printf("CheckCapacity:%s", strerror(errno));return 0;}else{pc->data = ptr;pc->capacity += INC_SZ;printf("增容成功,当前容量:%d\n", pc->capacity);return 1;}}return 1;
}
//*****************************************
//增加指定联系人
void AddContact(Contact* pc)
{if (0 == CheckCapacity(pc)){printf("空间不够,扩容失败\n");return;}else{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");}
}

运行截图:
在这里插入图片描述

2.3、删除联系人信息

首先我们先要输入要删除的人的名字,查看此人是否存在(将此功能单独封装成一个函数,因为后续查询指定联系人功能也要用到)。
查询到要被删除的人存在,则调用DelContact函数进行删除。
我们所说的删除,并不是直接把数据在内存中删掉,而是用后面的数据前移或后移进行覆盖,达到删除效果。

在这里插入图片描述

//查找
static int FindByName(const Contact* pc, char name[])
{int i = 0;for (i = 0; i < pc->sz; i++){if (0 == strcmp(pc->data[i].name, name)){return i;}}return -1;}void DelContact(pContact pc)
{char name[MAX_NAME] = { 0 };if (pc->sz == 0){printf("通讯录为空,无法删除\n");return;}//删除//1.找到要删除的人 - 位置(下标)printf("输入要删除人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要删除的人不存在\n");return;}//2.删除 - 删除pos位置上的数据int i = 0;for (i = pos; i < pc->sz-1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除成功\n");
}

在这里插入图片描述

2.4、查询联系人信息

我们也需要调用FindByName函数,用户输入待查找联系人的名字,如果存在此人我们就将其信息打印出来。

代码分析:

void SearchContact(const Contact* pc)
{char name[MAX_NAME] = { 0 };printf("请输入要查找的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在\n");return;}//打印printf("%-10s %-4s %-5s %-12s %-30s\n","姓名", "年龄", "性别", "电话", "地址");printf("%-10s %-4d %-5s %-12s %-30s\n",pc->data[pos].name, pc->data[pos].age,pc->data[pos].sex, pc->data[pos].tele,pc->data[pos].addr);}

运行截图:
在这里插入图片描述

2.5、修改联系人信息

要修改联系人信息自然也要先调用FindByName函数,在通讯录中找到该联系人,如果找到了,我们就将该联系人信息重新录入一次就行了。

void ModifyContact(Contact* pc)
{char name[MAX_NAME] = { 0 };printf("请输入要修改人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的人不存在\n");return;}//修改printf("请输入名字:>");scanf("%s", pc->data[pos].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");
}

2.6、显示所有联系人信息

我们首先打印一排信息栏,接下来循环sz次(即联系人个数次),打印每个联系人的信息即可。

代码分析:

void ShowContact(const Contact* pc)
{int i = 0;printf("%-10s %-4s %-5s %-12s %-30s\n","姓名","年龄","性别","电话","地址");for (i = 0; i < pc->sz; i++){printf("%-10s %-4d %-5s %-12s %-30s\n",pc->data[i].name, pc->data[i].age,pc->data[i].sex, pc->data[i].tele,pc->data[i].addr);}
}

运行截图:
在这里插入图片描述

2.7、对所有联系人信息进行排序整理

通过名字排序通讯录中联系人的先后顺序。
我们可以借助于已有的qosrt函数进行排序,只需要写一个自定义的比较函数即可。

//按照名字来排序
int cmp_by_name(void* e1, void* e2)
{return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}void SortContact(Contact* pc)
{qsort(pc->data, pc->sz, sizeof(PeoInfo), cmp_by_name);printf("排序成功\n");
}

2.8、删除所有联系人信息

因为我们使用动态内存函数开辟空间,所以需要使用free进行内存释放,并把data指针置NULL。

void DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->capacity = 0;pc->sz = 0;printf("释放内存......\n");
}

三、通讯录代码的整合

由于代码量相对来说有一点多,所以我们就将函数的声明的定义分开,这样有利于提高代码的可读性,同时会保持一个良好的思路,且方便编写代码。我们将函数的声明放在单独的一个contact.h的头文件,函数的实现放在一个单独的contact.c源文件,函数的主方法及调用放在另一个单独的test.c源文件。

在这里插入图片描述

源代码:

Contact.h文件

#include<stdio.h>
#include<string.h>
#include<stdlib.h>#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
#define DEFAULT_SZ 3
#define INC_SZ 2//表示一个人的信息
typedef struct PeoInfo
{char name[MAX_NAME];int age;char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];
}PeoInfo;//动态版本
//1.默认能够存放3个人的信息
//2.不够的话,每次增加2个人信息typedef struct Contact
{PeoInfo* data;//data指向了存放数据的空间int sz;//记录通讯录中有效信息个数int capacity;//通讯录当前的容量}Contact, * pContact;//初始化通讯录
void InitContact(Contact* pc);//销毁通讯录
void DestroyContact(Contact* pc);//增加指定联系人
void AddContact(Contact* pc);//显示联系人信息
void ShowContact(const Contact* pc);//删除指定联系人
void DelContact(pContact pc);//查找指定联系人
void SearchContact(const Contact* pc);//修改通讯录
void ModifyContact(Contact* pc);//排序
void SortContact(Contact* con);

Contact.c文件

#include"contact.h"void InitContact(Contact* pc)
{pc->data = (PeoInfo*)malloc(DEFAULT_SZ * sizeof(PeoInfo));if (pc->data == NULL){printf("通讯录初始化失败:%s\n",strerror(errno));return;}pc->sz = 0;pc->capacity = DEFAULT_SZ;}void DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->capacity = 0;pc->sz = 0;printf("释放内存......\n");
}static int CheckCapacity(Contact* pc)//扩容
{if (pc->sz == pc->capacity){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeoInfo));if (ptr == NULL){printf("CheckCapacity:%s", strerror(errno));return 0;}else{pc->data = ptr;pc->capacity += INC_SZ;printf("增容成功,当前容量:%d\n", pc->capacity);return 1;}}return 1;
}void AddContact(Contact* pc)
{if (0 == CheckCapacity(pc)){printf("空间不够,扩容失败\n");return;}else{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");}
}void ShowContact(const Contact* pc)
{int i = 0;printf("%-10s %-4s %-5s %-12s %-30s\n","姓名","年龄","性别","电话","地址");for (i = 0; i < pc->sz; i++){printf("%-10s %-4d %-5s %-12s %-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(const Contact* pc, char name[])
{int i = 0;for (i = 0; i < pc->sz; i++){if (0 == strcmp(pc->data[i].name, name)){return i;}}return -1;}void DelContact(pContact pc)
{char name[MAX_NAME] = { 0 };if (pc->sz == 0){printf("通讯录为空,无法删除\n");return;}//删除//1.找到要删除的人 - 位置(下标)printf("输入要删除人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要删除的人不存在\n");return;}//2.删除 - 删除pos位置上的数据int i = 0;for (i = pos; i < pc->sz-1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除成功\n");
}void SearchContact(const Contact* pc)
{char name[MAX_NAME] = { 0 };printf("请输入要查找的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在\n");return;}//打印printf("%-10s %-4s %-5s %-12s %-30s\n","姓名", "年龄", "性别", "电话", "地址");printf("%-10s %-4d %-5s %-12s %-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)
{char name[MAX_NAME] = { 0 };printf("请输入要修改人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的人不存在\n");return;}//修改printf("请输入名字:>");scanf("%s", pc->data[pos].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");
}//按照名字来排序
int cmp_by_name(void* e1, void* e2)
{return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}void SortContact(Contact* pc)
{qsort(pc->data, pc->sz, sizeof(PeoInfo), cmp_by_name);printf("排序成功\n");
}

test.c文件

#include"contact.h"void menu()
{printf("=================================\n");printf("======       通讯录        ======\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");}enum Option
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};int main()
{int input = 0;Contact con;//通讯录//初始化通讯录InitContact(&con);do{menu();printf("请选择:>");scanf("%d", &input);switch (input){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:DestroyContact(&con);printf("退出通讯录\n");break;default:printf("选择错误\n");break;}} while (input);return 0;
}

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

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

相关文章

[极客大挑战 2019]BuyFlag 1(两种解法)

题目环境&#xff1a; FLAG NEED YOUR 100000000 MONEY flag需要你的100000000元 F12瞅瞅源代码&#xff1a; if (isset($_POST[password])){ $password $_POST[password]; if (is_numeric($password)) { echo "password cant be number" } elseif ($pas…

【MySQL】列属性

文章目录 CHAR和VARCHAR插入单行 INSERT INTO插入多行插入分层行 LAST_INSERT_IN()创建表复制 CREAT TABLE AS更新单行 UPDATE...SET更新多行在UPDATES中使用子查询【需着重复习】删除行 DELETE恢复数据库到原始状态 CHAR和VARCHAR CHAR(50)&#xff1a;存储文本占5个字符&…

Git 命令详解

系列文章目录 C高性能优化编程系列 深入理解软件架构设计系列 高级C并发线程编程 C技能系列 期待你的关注哦&#xff01;&#xff01;&#xff01; 现在的一切都是为将来的梦想编织翅膀&#xff0c;让梦想在现实中展翅高飞。 Now everything is for the future of dream we…

【博士每天一篇文献-算法】Imposing Connectome-Derived Topology on an Echo State Network

阅读时间&#xff1a;2023-11-5 1 介绍 年份&#xff1a;2022 作者&#xff1a;Jacob Morra, Mark Daley 西部大学 期刊&#xff1a;2022 International Joint Conference on Neural Networks (IJCNN) 引用量&#xff1a;3 研究了果蝇连接图的拓扑结构对混沌时间序列预测中回…

MGA-WPA

作者未提供代码

YOLO目标检测——猫狗目标检测数据集下载分享【含对应voc、coco和yolo三种格式标签】

实际项目应用&#xff1a;宠物识别、猫狗分类数据集说明&#xff1a;猫狗分类检测数据集&#xff0c;真实场景的高质量图片数据&#xff0c;数据场景丰富&#xff0c;含有猫和狗图片标签说明&#xff1a;使用lableimg标注软件标注&#xff0c;标注框质量高&#xff0c;含voc(xm…

【开源】基于Vue.js的智能停车场管理系统的设计和实现

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容A. 车主端功能B. 停车工作人员功能C. 系统管理员功能1. 停车位模块2. 车辆模块3. 停车记录模块4. IC卡模块5. IC卡挂失模块 三、界面展示3.1 登录注册3.2 车辆模块3.3 停车位模块3.4 停车数据模块3.5 IC卡档案模块3.6 IC卡挂…

Spring Boot中使用Spring Data JPA访问MySQL

Spring Data JPA是Spring框架提供的用于简化JPA&#xff08;Java Persistence API&#xff09;开发的数据访问层框架。它通过提供一组便捷的API和工具&#xff0c;简化了对JPA数据访问的操作&#xff0c;同时也提供了一些额外的功能&#xff0c;比如动态查询、分页、排序等。 …

ThinkPHP图片处理之压缩图片大小,图片处理之图片水印(添加平铺文字水印,并设置文字之间的间距和文字的角度)

安装扩展 使用Composer安装图像处理类库 composer require topthink/think-image在对应的控制器内引入Image use think\Image;图片处理之压缩图片大小 public function upload(){$file request()->file(image);// 将前端传过来的图片移动到项目目录下$info $file->…

PostgreSQL 技术内幕(十一)位图扫描

扫描算子在上层计算和底层存储之间&#xff0c;向下扫描底层存储的数据&#xff0c;向上作为计算的输入源&#xff0c;在SQL的执行层中&#xff0c;起着关键的作用。顺序、索引、位图等不同类型的扫描算子适配不同的数据分布场景。然而&#xff0c;扫描算子背后的实现原理是怎样…

分布式理论基础:CAP定理

什么是CAP CAP原则又称CAP定理&#xff0c;指的是在一个分布式系统中&#xff0c;Consistency&#xff08;一致性&#xff09;、 Availability&#xff08;可用性&#xff09;、Partition tolerance&#xff08;分区容错性&#xff09;这三个基本需求&#xff0c;最多只能同时…

【数据结构】:红黑树

1、红黑树的简介 红黑树&#xff08;Red Black Tree&#xff09; 是一种自平衡二叉查找树&#xff0c;是在计算机科学中用到的一种数据结构。 红黑树是在1972年由Rudolf Bayer发明的&#xff0c;当时被称为平衡二叉B树&#xff08;symmetric binary B-trees&#xff09;。后来…