模拟通讯录(详解通讯录排序qsort,strcmp)

2024-06-03 05:28

本文主要是介绍模拟通讯录(详解通讯录排序qsort,strcmp),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:

        学习了C语言结构体、联合体、枚举等,就可以写一个通讯录来强化自己对结构体的理解学习。顺便提升大家的基本功!!

通讯录菜单的打印:

        关于菜单的打印在之前写游戏的时候写过多次,大家可以参照之前的改写菜单。

代码如下:

int main()
{int input = 0;do{menu();printf("请输入你的选择:>");scanf("%d", &input);switch(input){case 1:break;case 2:break;case 3:break;case 4:break;case 5:break;case 6:break;case 0:printf("退出通讯录\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);return 0;
}
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");
}

关于菜单的打印在这里不再进行强调。

        如果大家在switch的时候选择太多,都是数字不知道是什么意思,此时大家可以利用枚举常量,把这些数字与含义联系到一起。

        EXIT        表示0;

        ADD        表示1;

        DEL        表示2;

        SEARCH 表示 3;

        MODIFY  表示4;

        SHOW    表示5;

         SORT     表示6;        

        代码如下:

        

enum Option
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT,
};
int main()
{int input = 0;do{menu();switch (input){case ADD:break;case DEL:break;case SEARCH:break;case MODIFY:break;case SHOW:break;case SORT:break;case EXIT:break;default:break;}} while (input);return 0;
}

定义并初始化通讯录:

定义通讯录:

        菜单打印完成后,就需要定义通讯录,并且要将其初始化为0,方便后期增加联系人

将定义的一些函数或者是量放在.h的头文件中。

#define NAME_MAX 20
#define SEX_MAX 10
#define TELE_MAX 12
#define ADDR_MAX 30
#define MAX 100
//定义一个通讯录
typedef struct PeoInfo
{char name[NAME_MAX];int age;char sex[SEX_MAX];char tele[TELE_MAX];char addr[ADDR_MAX];
}PeoInfo;
//定义整个通讯录
typedef struct Contact
{PeoInfo data[MAX];int sz;//记录通信录中人的信息个数
}Contact;

初始化通讯录:

        我们可以再创建一个源文件,在创建好的源文件中初始化。

void InitContaxt(Contact* pc)//初始化通讯录
{pc->sz = 0;memset(pc->data, 0, sizeof(pc->data));
}

这里用到了memset库函数,可以进行对内存块的赋值。

添加联系人:

        在contact源文件定义addcontact函数,在.h文件中声明。在1.文件中使用。

代码如下:

            开始前必须判断有没有空间放得下。(用if语句)

void addcontact(Contact* pc)
{if (pc->sz == MAX){printf("通讯录已满,无法存放\n");return;}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);printf("添加成功\n");pc->sz++;
}

展示通讯录:

        showcontact函数。

void showcontact(Contact* pc)
{if (pc->sz == 0){printf("通讯录为空\n");return;}printf("%-20s%-5s%-5s%-20s%-30s\n", "姓名:", "年龄:", "性别:", "电话:", "住址");int i = 0;for (i = 0; i < pc->sz; i++){printf("%-20s%-5d%-5s%-20s%-30s\n", pc->data[i].name, pc->data[i].age,pc->data[i].sex, pc->data[i].tele, pc->data[i].addr);}
}

删除联系人:

        注意事项:

        1、删除之前需要判断该联系人存不存在!!

        2、删除可以理解为覆盖,用删除的联系人之后的覆盖前面的。

        3、两个字符串比较大小必须用strcmp库函数。

        4、可以把查找联系人存不存在封装成一个函数。

void delcontact(Contact* pc)
{char name[NAME_MAX];assert(pc);printf("请输入你要删除联系人的姓名\n");scanf("%s",name);if (pc->sz == 0){printf("通讯录为空,无法删除\n");return;}int ret = Find(pc, name);if (ret == -1){printf("无法找到该联系人\n");return;}else{int i = 0;for (i = ret; i < pc->sz-1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除成功\n");}
}

查找联系人的find函数代码如下:

        

int Find(Contact* pc,char*name)
{int i = 0;for (i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, name) == 0){return i;}}return -1;
}

查找联系人:

        searchcontact函数:

void searchcontact(Contact* pc)
{char name[NAME_MAX];assert(pc);printf("请输入你要删除联系人的姓名\n");scanf("%s", name);if (pc->sz == 0){printf("通讯录为空,无法查找\n");return;}int ret = Find(pc, name);if (ret == -1){printf("无法找到该联系人\n");return;}else{printf("%-20s%-5s%-5s%-20s%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("%-20s%-5d%-5s%-20s%-30s\n", pc->data[ret].name, pc->data[ret].age, pc->data[ret].sex, pc->data[ret].tele, pc->data[ret].addr);printf("查找成功\n");}}

修改联系人信息:

void modifycontact(Contact* pc)
{char name[NAME_MAX];assert(pc);if (pc->sz == 0){printf("通讯录为空,无法修改\n");return;}printf("输入要修改的人的名字:");scanf("%s", name);int ret = Find(pc, name);if (ret == -1){printf("要修改的人不存在\n");return;}printf("输入名字:");scanf("%s", pc->data[ret].name);printf("输入年龄:");scanf("%d", &pc->data[ret].age);printf("输入性别:");scanf("%s", pc->data[ret].sex);printf("输入电话:");scanf("%s", pc->data[ret].tele);printf("输入地址:");scanf("%s", pc->data[ret].addr);
}

*联系人排序:

利用strcmp函数冒泡排序:

        要点:

        注意结构体之间的交换。
        注意strcmp函数的使用,如果不懂,可以看写文章-CSDN创作中心中有介绍。

void sortcontact(Contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空,无法排序\n");return;}int i = 0;for (i = 0; i < pc->sz; i++){int j = 0;for (j = 0; j < pc ->sz - i-1; j++){if ((strcmp(pc->data[j].name, pc->data[j + 1].name))>0){//交换结构体PeoInfo a = pc->data[j];pc->data[j] = pc->data[j + 1];pc->data[j + 1] = a;}}}printf("排序成功\n");
}

利用qsort函数快速排序:

        要点:

        如果想使用qsort函数快速排序,必须熟知qsort函数的使用原理,它的4个参数分别是什么,分别表示什么意思,最好是自己模拟过qsor函数,用起来更加方便!!

        不懂qsort函数的使用可以看C语言指针详解(3)-CSDN博客,里面有相关qsort的介绍和模拟实现。

        代码如下:

        

int compar(const void *p1,const void *p2)
{return *(char*)p1 - *(char*)p2;
}void sortcontact(Contact* pc)
{
assert(pc);if (pc->sz == 0){printf("通讯录为空,无法排序\n");return;}qsort(pc->data->name, pc->sz, sizeof(PeoInfo), compar);
}

注:用qsort确实简单,但是必须注意qsort的参数分别表示什么含义,注意compar函数的自定义!!

这篇关于模拟通讯录(详解通讯录排序qsort,strcmp)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1026115

相关文章

十四、观察者模式与访问者模式详解

21.观察者模式 21.1.课程目标 1、 掌握观察者模式和访问者模式的应用场景。 2、 掌握观察者模式在具体业务场景中的应用。 3、 了解访问者模式的双分派。 4、 观察者模式和访问者模式的优、缺点。 21.2.内容定位 1、 有 Swing开发经验的人群更容易理解观察者模式。 2、 访问者模式被称为最复杂的设计模式。 21.3.观察者模式 观 察 者 模 式 ( Obser

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

Jitter Injection详解

一、定义与作用 Jitter Injection,即抖动注入,是一种在通信系统中人为地添加抖动的技术。该技术通过在发送端对数据包进行延迟和抖动调整,以实现对整个通信系统的时延和抖动的控制。其主要作用包括: 改善传输质量:通过调整数据包的时延和抖动,可以有效地降低误码率,提高数据传输的可靠性。均衡网络负载:通过对不同的数据流进行不同程度的抖动注入,可以实现网络资源的合理分配,提高整体传输效率。增

Steam邮件推送内容有哪些?配置教程详解!

Steam邮件推送功能是否安全?如何个性化邮件推送内容? Steam作为全球最大的数字游戏分发平台之一,不仅提供了海量的游戏资源,还通过邮件推送为用户提供最新的游戏信息、促销活动和个性化推荐。AokSend将详细介绍Steam邮件推送的主要内容。 Steam邮件推送:促销优惠 每当平台举办大型促销活动,如夏季促销、冬季促销、黑色星期五等,用户都会收到邮件通知。这些邮件详细列出了打折游戏、

探索Elastic Search:强大的开源搜索引擎,详解及使用

🎬 鸽芷咕:个人主页  🔥 个人专栏: 《C++干货基地》《粉丝福利》 ⛺️生活的理想,就是为了理想的生活! 引入 全文搜索属于最常见的需求,开源的 Elasticsearch (以下简称 Elastic)是目前全文搜索引擎的首选,相信大家多多少少的都听说过它。它可以快速地储存、搜索和分析海量数据。就连维基百科、Stack Overflow、

数据结构9——排序

一、冒泡排序 冒泡排序(Bubble Sort),顾名思义,就是指越小的元素会经由交换慢慢“浮”到数列的顶端。 算法原理 从左到右,依次比较相邻的元素大小,更大的元素交换到右边;从第一组相邻元素比较到最后一组相邻元素,这一步结束最后一个元素必然是参与比较的元素中最大的元素;按照大的居右原则,重新从左到后比较,前一轮中得到的最后一个元素不参4与比较,得出新一轮的最大元素;按照上述规则,每一轮结

基于 Java 实现的智能客服聊天工具模拟场景

服务端代码 import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;import java.io.PrintWriter;import java.net.ServerSocket;import java.net.Socket;public class Serv

常用MQ消息中间件Kafka、ZeroMQ和RabbitMQ对比及RabbitMQ详解

1、概述   在现代的分布式系统和实时数据处理领域,消息中间件扮演着关键的角色,用于解决应用程序之间的通信和数据传递的挑战。在众多的消息中间件解决方案中,Kafka、ZeroMQ和RabbitMQ 是备受关注和广泛应用的代表性系统。它们各自具有独特的特点和优势,适用于不同的应用场景和需求。   Kafka 是一个高性能、可扩展的分布式消息队列系统,被设计用于处理大规模的数据流和实时数据传输。它

七种排序方式总结

/*2018.01.23*A:YUAN*T:其中排序算法:冒泡排序,简单排序,直接插入排序,希尔排序,堆排序,归并排序,快速排序*/#include <stdio.h>#include <math.h>#include <malloc.h>#define MAXSIZE 10000#define FALSE 0#define TRUE 1typedef struct {i

Linux中拷贝 cp命令中拷贝所有的写法详解

This text from: http://www.jb51.net/article/101641.htm 一、预备  cp就是拷贝,最简单的使用方式就是: cp oldfile newfile 但这样只能拷贝文件,不能拷贝目录,所以通常用: cp -r old/ new/ 那就会把old目录整个拷贝到new目录下。注意,不是把old目录里面的文件拷贝到new目录,