C语言结构体小项目之通讯录代码实现+代码分析

一、思路

1.文件

这里由于通讯录实现代码较长,因此分三个文件进行,contact.c用于实现通讯录主体代码,声明各项头文件用contact.h实现,测试用test.c

二.功能

  1. 增加联系人
  2. 删除联系人
  3. 修改联系人
  4. 查找指定联系人
  5. 排序
  6. 显示通讯录的信息

三.分步实现代码

1.初始化结构体

通讯录中每个联系人的信息至少包括:姓名、年龄、性别、电话、住址。
我们可以将这些内容定义为一个结构体,因为每个联系人的信息都有这几个要素:

#define NAME_MAX 20
#define SEX_MAX 20
#define TELE_MAX 20
#define ADDR_MAX 20typedef struct PeoInfo
{char name[NAME_MAX];int age;char sex[SEX_MAX];char tele[TELE_MAX];char addr[ADDR_MAX];
}PeoInfo;
typedef是定义结构体类型由struct PeoInfo---> PeoInfo (这样后面使用定义变量时类型就直接写PeoInfo就可以了)
数组【】里面如果是常量 那就后期想修改是非常不方便的,所以我们自己定义NAME_MAX等,后期修改起来更加方便。

但当我们操作通讯录联系人时,需要知道联系人是否达到上限,如果达到则无法再添加联系人

因此我们将联系人的信息与通讯录已有联系人个数联系,一同放入结构体,避免后续功能需同时传入两个变量

#define MAX 100
typedef struct Contact
{PeoInfo data[MAX];//存放数据int sz;//记录当前通讯录中存放的人的信息个数
}Contact;//通讯录

因此后续访问通讯录信息时,即可通过contact访问data然后访问信息。

2.InitContact初始化个人信息的声明

//初始化通讯录
void InitContact(struct Contact* ps)
{memset(ps->data, 0, sizeof(ps->data));//将待用内存空间设置为0ps->size = 0;//设置通讯录最初只有0个元素
}

3.添加联系人的信息

最初应该通讯录判断已有联系人是否达到上线即用ps->size,达到则无法添加,否则就输入待添加联系人的信息,最后添加完成size++;

//添加一个联系人的信息
void AddContact(struct Contact* ps)
{if (ps->size == 1000){printf("通讯录已满,无法添加!\n");}//通讯录内联系人已达上限else{printf("请输入姓名:>");scanf("%s", ps->data[ps->size].name);printf("请输入年龄:>");scanf("%d", &(ps->data[ps->size].age));printf("请输入性别:>");scanf("%s", ps->data[ps->size].sex);printf("请输入电话:>");scanf("%s", ps->data[ps->size].tele);printf("请输入地址:>");scanf("%s", ps->data[ps->size].address);ps->size++;//通讯录中联系人个数+1printf("添加成功\n");}
}

4.删除联系人的信息

我们需要用户输入待删除联系人的名字,并在通讯录中查找是否存在该名字的联系人。(因为后面还会有此相同的操作,所以将其封装为一个函数,避免代码冗余),删除完成就size--

接着就是实现删除,我们只需通过输入字符串名字来找到该联系人所在下标,记录下来,然后来个for循环从后往前覆盖,并且防止越界j应该小于size-1。

int FindByName(char name[15],const struct Contact* ps)
{int i = 0;for (i = 0; i < ps->size; i++){if (strcmp(ps->data[i].name, name) == 0)return i;//找到了返回下标}return -1;//找不到的情况
}//删除一个联系人的信息
void DeleteContact(struct Contact* ps)
{char name[15];printf("请输入要删除联系人的姓名:>");scanf("%s", name);//查找int pos = FindByName(name, ps);//找到了返回下标,没找到返回-1//删除if (pos == -1){printf("未找到该联系人\n");}else{int j = 0;for (j = pos; j < ps->size - 1; j++){ps->data[j] = ps->data[j + 1];}//从要删除的联系人位置开始,后一位联系人信息覆盖前一个联系人信息printf("删除成功\n");ps->size--;//通讯录中联系人个数-1}
}

5.查找指定联系人并打印其信息

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

//查找指定联系人并打印其信息
void SreachContact(const struct Contact* ps)
{char name[15];printf("请输入要查找联系人的姓名:>");scanf("%s", name);int pos = FindByName(name, ps);//找到了返回下标,没找到返回-1if (pos == -1){printf("查无此人\n");}else{printf("%-15s\t%-4s\t%-5s\t%-12s\t%-20s\n", "姓名", "年龄", "性别", "电话", "住址");printf("%-15s\t%-4d\t%-5s\t%-12s\t%-20s\n",ps->data[pos].name,ps->data[pos].age,ps->data[pos].sex,ps->data[pos].tele,ps->data[pos].address);}//打印该联系人的信息内容
}

6.修改指定联系人的信息

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

//修改指定联系人的信息
void ModifyContact(struct Contact* ps)
{char name[15];printf("请输入要修改的联系人的姓名:>");scanf("%s", name);int pos = FindByName(name, ps);//找到了返回下标,没找到返回-1if (pos == -1){printf("查无此人\n");}else{printf("请输入姓名:>");scanf("%s", ps->data[pos].name);printf("请输入年龄:>");scanf("%d", &(ps->data[pos].age));printf("请输入性别:>");scanf("%s", ps->data[pos].sex);printf("请输入电话:>");scanf("%s", ps->data[pos].tele);printf("请输入地址:>");scanf("%s", ps->data[pos].address);printf("修改成功\n");}//修改联系人信息,即将该联系人信息重新录入
}

7.打印通讯录中的全部信息

首先判断通讯录是否为空(如果没有联系人,那就没有打印的必要了)。如果通讯录不为空,我们首先打印一排信息栏,接下来循环size次(即联系人个数次),打印每个联系人的信息即可。

//打印通讯录中的全部信息
void ShowContact(const struct Contact* ps)
{if (ps->size == 0){printf("通讯录为空\n");}//通讯录中联系人个数为0else{printf("%-15s\t%-4s\t%-5s\t%-12s\t%-20s\n", "姓名", "年龄", "性别", "电话", "住址");//打印信息栏int i = 0;for (i = 0; i < ps->size; i++){printf("%-15s\t%-4d\t%-5s\t%-12s\t%-20s\n",ps->data[i].name,ps->data[i].age,ps->data[i].sex,ps->data[i].tele,ps->data[i].address);}//打印通讯录全部信息内容}
}

8.通过名字排序通讯录中联系人的先后顺序

这个排序功能我们可以借助于已有的qosrt函数进行排序,只需要写一个自定义的比较函数即可。

//自定义的比较函数
int CmpByName(const void* e1, const void* e2)
{return strcmp((const char*)e1, (const char*)e2);
}//通过名字排序通讯录中联系人的先后顺序
void SortContact(struct Contact* ps)
{qsort(ps->data, ps->size, sizeof(struct PeoInfo), CmpByName);//排序
}

四.整体代码

contact.h

#define _CRT_SECURE_NO_WARNINGS 1#define MAX 1000#define MAX_NAME 15
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDRESS 20#include <stdio.h>
#include <string.h>
#include <stdlib.h>enum Option
{EXIT,//0ADD,//1DELETE,//2SREACH,//3MODIFY,//4SHOW,//5SORT//6
};//增加代码可读性struct PeoInfo
{char name[MAX_NAME];int age;char sex[MAX_SEX];int tele[MAX_TELE];char address[MAX_ADDRESS];
};//每一个联系人的信息内容//通讯录类型
struct Contact
{struct PeoInfo data[MAX];//存放1000个联系人的信息int size;//记录当前已有联系人的个数
};//初始化通讯录
void InitContact(struct Contact* ps);//添加一个联系人的信息
void AddContact(struct Contact* ps);//删除一个联系人的信息
void DeleteContact(struct Contact* ps);//查找指定联系人并打印其信息
void SreachContact(const struct Contact* ps);//修改指定联系人的信息
void ModifyContact(struct Contact* ps);//打印通讯录中的全部信息
void ShowContact(const struct Contact* ps);//通过名字排序通讯录中联系人的先后顺序
void SortContact(struct Contact* ps);

contact.c

#include "contact.h"//初始化通讯录
void InitContact(struct Contact* ps)
{memset(ps->data, 0, sizeof(ps->data));//将待用内存空间设置为0ps->size = 0;//设置通讯录最初只有0个元素
}//添加一个联系人的信息
void AddContact(struct Contact* ps)
{if (ps->size == MAX){printf("通讯录已满,无法添加!\n");}//通讯录内联系人已达上限else{printf("请输入姓名:>");scanf("%s", ps->data[ps->size].name);printf("请输入年龄:>");scanf("%d", &(ps->data[ps->size].age));printf("请输入性别:>");scanf("%s", ps->data[ps->size].sex);printf("请输入电话:>");scanf("%s", ps->data[ps->size].tele);printf("请输入地址:>");scanf("%s", ps->data[ps->size].address);ps->size++;//通讯录中联系人个数+1printf("添加成功\n");}
}//通过名字查找联系人,找到了返回下标,没找到返回-1
static int FindByName(char name[MAX_NAME],const struct Contact* ps)
{int i = 0;for (i = 0; i < ps->size; i++){if (strcmp(ps->data[i].name, name) == 0)return i;//找到了返回下标}return -1;//找不到的情况
}//删除一个联系人的信息
void DeleteContact(struct Contact* ps)
{char name[MAX_NAME];printf("请输入要删除联系人的姓名:>");scanf("%s", name);//查找int pos = FindByName(name, ps);//找到了返回下标,没找到返回-1//删除if (pos == -1){printf("查无此人\n");}else{int j = 0;for (j = pos; j < ps->size - 1; j++){ps->data[j] = ps->data[j + 1];}//从要删除的联系人位置开始,后一位联系人信息覆盖前一个联系人信息printf("删除成功\n");ps->size--;//通讯录中联系人个数-1}
}//查找指定联系人并打印其信息
void SreachContact(const struct Contact* ps)
{char name[MAX_NAME];printf("请输入要查找联系人的姓名:>");scanf("%s", name);int pos = FindByName(name, ps);//找到了返回下标,没找到返回-1if (pos == -1){printf("查无此人\n");}else{printf("%-15s\t%-4s\t%-5s\t%-12s\t%-20s\n", "姓名", "年龄", "性别", "电话", "住址");printf("%-15s\t%-4d\t%-5s\t%-12s\t%-20s\n",ps->data[pos].name,ps->data[pos].age,ps->data[pos].sex,ps->data[pos].tele,ps->data[pos].address);}//打印该联系人的信息内容
}//修改指定联系人的信息
void ModifyContact(struct Contact* ps)
{char name[MAX_NAME];printf("请输入要修改的联系人的姓名:>");scanf("%s", name);int pos = FindByName(name, ps);//找到了返回下标,没找到返回-1if (pos == -1){printf("查无此人\n");}else{printf("请输入姓名:>");scanf("%s", ps->data[pos].name);printf("请输入年龄:>");scanf("%d", &(ps->data[pos].age));printf("请输入性别:>");scanf("%s", ps->data[pos].sex);printf("请输入电话:>");scanf("%s", ps->data[pos].tele);printf("请输入地址:>");scanf("%s", ps->data[pos].address);printf("修改成功\n");}//修改联系人信息,即将该联系人信息重新录入
}//打印通讯录中的全部信息
void ShowContact(const struct Contact* ps)
{if (ps->size == 0){printf("通讯录为空\n");}//通讯录中联系人个数为0else{printf("%-15s\t%-4s\t%-5s\t%-12s\t%-20s\n", "姓名", "年龄", "性别", "电话", "住址");//打印信息栏int i = 0;for (i = 0; i < ps->size; i++){printf("%-15s\t%-4d\t%-5s\t%-12s\t%-20s\n",ps->data[i].name,ps->data[i].age,ps->data[i].sex,ps->data[i].tele,ps->data[i].address);}//打印通讯录全部信息内容}
}//自定义的比较函数
int CmpByName(const void* e1, const void* e2)
{return strcmp((const char*)e1, (const char*)e2);
}//通过名字排序通讯录中联系人的先后顺序
void SortContact(struct Contact* ps)
{qsort(ps->data, ps->size, sizeof(struct PeoInfo), CmpByName);//排序
}

test.c

#include "contact.h"void menu()
{printf("|-----------------------|\n");printf("|        contact        |\n");printf("|   1.Add     2.Delete  |\n");printf("|   3.Search  4.Modify  |\n");printf("|   5.Show    6.Sort    |\n");printf("|        0.Exit         |\n");printf("|-----------------------|\n");
}//打印菜单int main()
{int input = 0;//创建通讯录struct Contact con;//con就是通讯录,里面包含1000个元素的数组和size//初始化通讯录InitContact(&con);do{menu();printf("请选择:>");scanf("%d", &input);switch (input){case ADD:AddContact(&con);break;case DELETE:DeleteContact(&con);break;case SREACH:SreachContact(&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;
}

最后,运用枚举可以增加代码可读性,因为你如果用1,2,3等用户可能不明白意思。但是用枚举实现,既符合你输入习惯,又规范

用宏定义来定义一些数据,是为了方便后续修改。

 

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

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

相关文章

Mac 打不开github解决方案

序言 github 时有打不开的情况&#xff0c;为此很是烦恼&#xff0c;这里分享一下如何解决这种问题&#xff0c;其实问题的本质是在访问github网页时无法通过github.com的二级域名进行动态域名解析。 解决方案 手动配置静态文件hosts&#xff0c;将该域名和IP的映射关系添加…

vue文件下载请求blob文件流token失效的问题

页面停留很久token失效没有刷新页面&#xff0c;这时候点击下载依然可以导出文件&#xff0c;但是文件打不开且接口实际上返回的是401&#xff0c;这是因为文件下载的方式通过window创建a标签的形式打开的&#xff0c;并没有判断token失效问题 const res await this.$axios.…

10:00面试,10:08就出来了,问的问题超出我认知

本来在上家公司上班&#xff0c;加班是每天必不可少的&#xff0c;但是看在加班费给的比较多的份上&#xff0c;就没有太计较了。没想到9月份下一份通知&#xff0c;所有人不准加班&#xff0c;加班费不仅没有了&#xff0c;薪资还要降30%,这下搞的生活都生活不下去了。 还好有…

Dijkstra求最短路 I(Dijkstra算法)

给定一个 n 个点 m 条边的有向图&#xff0c;图中可能存在重边和自环&#xff0c;所有边权均为正值。 请你求出 1 号点到 n 号点的最短距离&#xff0c;如果无法从 1 号点走到 n 号点&#xff0c;则输出 −1。 输入格式 第一行包含整数 n 和 m。 接下来 m 行每行包含三个整…

vscode 常用 Emmet Abbreviation 快捷方式

vscode 常用 Emmet Abbreviation 快捷方式 输入快捷指令后&#xff0c; 按“tab”键或者回车键 即可 .box*5&#xff1a;生成 5 个 class 为 box 的 div 元素 <div class"box"></div> <div class"box"></div> <div class&quo…

python和pygame实现捉小兔游戏

python和pygame实现捉小兔游戏 python和pygame实现捉小兔游戏&#xff0c;需要安装使用第三方库pygame&#xff0c;关于Python中pygame游戏模块的安装使用可见 https://blog.csdn.net/cnds123/article/details/119514520 下面是使用Python和Pygame创建的游戏&#xff0c;其中有…

常用的Linux基本命令

这些是一些常用的Linux基本命令&#xff0c;涵盖了文件操作、系统管理、进程管理、磁盘管理等方面&#xff1a; ls&#xff1a;列出目录内容cd&#xff1a;切换当前工作目录pwd&#xff1a;显示当前工作目录的绝对路径mkdir&#xff1a;创建新目录rmdir&#xff1a;删除空目录…

Pytorch从零开始实战13

Pytorch从零开始实战——ResNet与DenseNet探索 本系列来源于365天深度学习训练营 原作者K同学 文章目录 Pytorch从零开始实战——ResNet与DenseNet探索环境准备数据集模型选择开始训练可视化总结 环境准备 本文基于Jupyter notebook&#xff0c;使用Python3.8&#xff0c;P…

Java连接数据库实现用户登录和注册功能

目录 需求内容如下 示例代码 数据库studb Java代码 效果图 需求内容如下 1&#xff0c;创建数据库studb 2&#xff0c;库中添加用户表userinfo,包含如下字段 用户id ,用户名&#xff0c;用户密码&#xff0c;用户权限 &#xff08;数据类型和约束自己定义&#xff09…

web微服务规划

一、背景 通过微服务来搭建web系统&#xff0c;就要对微服务进行规划&#xff0c;包括服务的划分&#xff0c;每个服务和数据库的命名规则&#xff0c;服务用到的端口等。 二、微服务划分 1、根据业务进行拆分 如&#xff1a; 一个购物系统可以将微服务拆分为基础中心、会员…

SpringMVC异常处理机制

2.1 异常描述 在J2EE项目的开发中&#xff0c;不管是对底层的数据库操作过程&#xff0c;还是业务层的处理过程&#xff0c;还是控制层的处理过程&#xff0c;都不可避免会遇到各种可预知的、不可预知的异常需要处理。每个过程都单独处理异常&#xff0c;系统的代码耦合度高&a…

【C++入门到精通】 线程库 | thread类 C++11 [ C++入门 ]

阅读导航 引言一、thread类的简单介绍二、线程函数详细介绍1. start() 函数&#xff08;1&#xff09;头文件&#xff08;2&#xff09;函数原型 2. join() 函数&#xff08;1&#xff09;头文件&#xff08;2&#xff09;函数原型 3. detach() 函数&#xff08;1&#xff09;头…

LeetCode Hot100 25.K个一组翻转链表

题目&#xff1a; 给你链表的头节点 head &#xff0c;每 k 个节点一组进行翻转&#xff0c;请你返回修改后的链表。 k 是一个正整数&#xff0c;它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍&#xff0c;那么请将最后剩余的节点保持原有顺序。 你不能只是单纯…

7+m6A+分型+实验,甲基化方向的生信思路,没有思路的同学可参考

今天给同学们分享一篇生信文章“Landscape analysis of m6A modification regulators related biological functions and immune characteristics in myasthenia gravis”&#xff0c;这篇文章发表在J Transl Med期刊上&#xff0c;影响因子为7.4。 结果解读&#xff1a; MG相…

快速文件搜索软件 Everything-voidtools

Everything 文件搜索软件 "Everything"是一款快速、轻巧且高效的桌面搜索引擎软件&#xff0c;专门用于在 Windows 操作系统上搜索文件和文件夹。它的主要特点是超快的搜索速度和精准的搜索结果&#xff0c;能够让用户快速找到他们需要的文件或文件夹。 1. 快速搜…

IDEA小技巧

目录 1. IDEA自动添加注释 创建类的时候自动添加注释 创建函数、方法的注释 1. IDEA自动添加注释 参考文档&#xff1a;idea java 自动添加文件注释 idea新建类自动注释_mob6454cc73c728的技术博客_51CTO博客 【操作工具】IDEA创建类及已有类添加注释-详细操作_idea设置创建…

搭建个人智能家居 开篇(搭建Home Assistant)

搭建个人智能家居 开篇&#xff08;搭建Home Assistant&#xff09; 前言Home Assistant搭建Home AssistantUbuntu系统搭建Windows系统搭建VM安装方法VirtualBox安装方法&#xff1a; 配置Home Assistant控制页面 前言 随着科技的进步、发展&#xff0c;物联网给我们的生活带来…

JdbcTemplate query系列方法指定jdbcType类型

使用SqlParameterValue类包装一下就行了&#xff0c;只要创建一个SqlParameterValue对象&#xff0c;通过构造函数把jdbcType类型&#xff08;用的是Types中的常量&#xff09;和值传入 例如&#xff1a; // 这两个包下面的 import org.springframework.jdbc.core.SqlParamete…

c YUV 转 JPEG(准备霍夫曼编码)

先取yuv 文件中一个168的块&#xff0c;跑通全流程 理解与思路&#xff1a; 1.块分割 YUV 文件分为&#xff1a;YUV444 YUV 422 YUV420。444:就是&#xff1a;12个char 有4个Y&#xff0c;4个U&#xff0c;4个 U&#xff0c;422&#xff1a;8个char 中有4个Y &#x…

react中使用antd的tabs进行标签切换时,销毁dom

前言&#xff1a;最近接的一个需求&#xff0c;需要将之前通过点击按钮触发跳转到的另外一个页面&#xff0c;通过跟点击按钮的页面融合&#xff0c;通过tab的形式进行切换。【由于业务方面的需求&#xff0c;两个页面的逻辑代码部分存在一致性&#xff08;经多年长期沿用下来的…