C语言——通讯录详解(动态版)

通讯录详解

  • 前言:
  • 一、定义一个通讯录
  • 二、初始化
  • 三、增加联系人
    • 3.1 给通讯录扩容
    • 3.2增加联系人
  • 四、释放内存
  • 五、完整代码

前言:

我们已经学过了通讯录的静态版,但是它的缺点很明显,通讯录满了就添加不了联系人了啦。我再让通讯录升级一下,写一个动态版通讯录。(通讯录静态版的超链接)
改造目标:
1、通讯录的空间不是固定,大小可以调整
2、默认能放3个人的信息,如果不够就增加,这里我一次增加3人。

一、定义一个通讯录

//定义一个通讯录
typedef struct Comcation
{com *data;//可以存100个人的信息int sz;//记录存放人的位置int capacity;//记录容量
}contact;

结构体增加了一个记录容量的成员capacity

二、初始化

默认通讯录能放三个人的信息

void initCantact(contact* pc)
{assert(pc);pc->data = (com*)malloc(3*sizeof(com));if (pc->data == NULL){perror("initCantact");return;}pc->sz = 0;pc->capacity = CONTACT_SZ;
}

让结构体初始化

三、增加联系人

3.1 给通讯录扩容

因为通讯录只能放三个人的信息,所以我们需要给通讯录扩容

int Addcapacity(contact* pc)
{if (pc->sz == pc->capacity)//容量满了就增容{com* ptr = (com*)realloc(pc->data, (pc->capacity + INT_sz) * sizeof(com));if (ptr == NULL){perror("Addcapacity");return 0;}else{pc->data = ptr;pc->capacity = pc->capacity + INT_sz;printf("增容成功\n");return 1;}}return 1;
}

返回的是个整型,然后我们根据返回的值判断是否增容失败。

3.2增加联系人

我在静态通讯录的基础上在多定义了两个宏
极大的方便了更改初始容量和一次增容的大小
在这里插入图片描述
然后就可以写增加联系人的函数

//增加联系人
void Addcontact(contact* pc)
{assert(pc);int n = Addcapacity(pc);if(0==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);pc->sz++;printf("成功增加联系人\n");
}

四、释放内存

因为我们开辟的内存是在堆上得,如果我们不手动释放,开辟的空间就会一直存在,然后消耗内存(内存泄漏)。
为了避免发生这种情况,在程序退出前,要释放内存。

//释放通讯录内存
void Destorycontacy(contact* pc)
{free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;
}

五、完整代码

在静态版的基础上只需改变上面的内容,就会成为通讯录的动态版。
具体的代码,如下:
cantact.h头文件(函数的声明和定义):

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <assert.h>
#define MAX 100//可以增加的人数
#define NAME_MAX 20//姓名
#define SEX_MAX 5//性别
#define TELE_MAX 12//电话
#define ADDR_MAX 20//地址
#define CONTACT_SZ 3//初始容量
#define INT_sz 3//一次增容的大小
//定义一个结构体
typedef struct Communication
{char name[NAME_MAX];//姓名int age;//年龄char sex[SEX_MAX];//性别char tele[TELE_MAX];//电话char addr[ADDR_MAX];//地址
}com;
//定义一个通讯录
typedef struct Comcation
{com *data;//可以存100个人的信息int sz;//记录存放人的位置int capacity;//记录容量
}contact;
//给通讯录赋初值
void initCantact(contact* pc);
//增容函数
int Addcapacity(contact* pc);
//菜单
void menu(void);
//增加联系人
void Addcontact(contact* pc);
//显示联系人
void Showcontact(const contact* pc);
//删除联系人
void Delcontact(contact* pc);
//查找联系人
int Findcontact(contact* pc, char arr[20]);
//查找指定联系人
void Searchcontact(contact* pc);
//修改联系人
void Modfycontatc(contact* pc);
//联系人按名字进行排序
void Sortcontact(contact* pc);
//清空联系人
void Emptycontact(contact* pc);
//释放内存
void Destorycontacy(contact* pc);

cantact.c源文件(函数的实现):

#include "cantact.h"
//通讯录初始化
void initCantact(contact* pc)
{assert(pc);pc->data = (com*)malloc(3*sizeof(com));if (pc->data == NULL){perror("initCantact");return;}pc->sz = 0;pc->capacity = CONTACT_SZ;
}
//增容函数
int Addcapacity(contact* pc)
{if (pc->sz == pc->capacity)//容量满了就增容{com* ptr = (com*)realloc(pc->data, (pc->capacity + INT_sz) * sizeof(com));if (ptr == NULL){perror("Addcapacity");return 0;}else{pc->data = ptr;pc->capacity = pc->capacity + INT_sz;printf("增容成功\n");return 1;}}return 1;
}
//增加联系人
void Addcontact(contact* pc)
{assert(pc);int n = Addcapacity(pc);if(0==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);pc->sz++;printf("成功增加联系人\n");
}
//显示联系人
void Showcontact(const contact* pc)
{assert(pc);printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n\n", "名字", "年龄", "性别", "电话", "地址");int i = 0;for (i = 0; i < pc->sz; i++){printf("%-20s\t%-4d\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);}
}
//查找联系人
int Findcontact(contact* pc, char arr[20])
{int i = 0;for (i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, arr) == 0)//找到了{return i;}}return -1;//没找到
}
//删除联系人
void Delcontact(contact* pc)
{if (pc->sz == 0){printf("没有联系人,无需删除\n");return;}char arr[20] = { 0 };assert(pc);printf("请输入要删除的人的名字:");scanf("%s", arr);//名字int n;n = Findcontact(pc, arr);if (n == -1){printf("没找到\n");return;}else//找到了{int i = 0;for (i = n; i < pc->sz - 1; i++)//从当前位置开始,把后面的值都往前面挪一位{pc->data[i] = pc->data[i + 1];}pc->sz--;}printf("成功删除联系人\n");
}
//查找指定联系人
void Searchcontact(contact* pc)
{assert(pc);char p[20] = { 0 };printf("请输入要查找人的姓名:");scanf("%s", p);int n = Findcontact(pc, p);if (n == -1){printf("要查找的人不存在\n");}else{printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");//打印数据printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[n].name,pc->data[n].age,pc->data[n].sex,pc->data[n].tele,pc->data[n].addr);}
}
//修改联系人
void Modfycontatc(contact* pc)
{assert(pc);char name[20] = { 0 };if (pc->sz == 0){printf("没有联系人,无需修改\n");return;}printf("请输入要修改人的名字;");scanf("%s", name);int n = 0;n = Findcontact(pc, name);if (n == -1){printf("要修改的人不存在\n");}else{printf("请输入修改后的名字:");scanf("%s", pc->data[n].name);printf("请输入修改后的年龄:");scanf("%d", &(pc->data[n].age));printf("请输入修改后的性别:");scanf("%s", pc->data[n].sex);printf("请输入修改后的电话号码:");scanf("%s", pc->data[n].tele);printf("请输入修改后的地址:");scanf("%s", pc->data[n].addr);printf("修改成功\n");}
}
int comper(const void* p1, const void* p2)//字符比较
{return strcmp(((com*)((contact*)p1)->data)->name, ((com*)((contact*)p2)->data)->name);
}
//按名字进行排序
void Sortcontact(contact* pc)
{qsort(pc, pc->sz, sizeof(pc->data[0]), comper);//快排
}
//清空联系人
void Emptycontact(contact* pc)
{assert(pc);if (pc->sz == 0){printf("联系人为空,无需清空\n");}else{pc->sz = 0;printf("清空成功\n");}
}
//释放通讯录内存
void Destorycontacy(contact* pc)
{free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;
}

test.c源文件(测试通讯录):

#include "cantact.h"
//菜单
void menu(void)
{printf("————————————————————\n");printf("|***************************************|\n");printf("|*****1.增加联系人     2.删除联系人*****|\n");printf("|*****3.修改联系人     4.查找联系人*****|\n");printf("|*****5.显示所有联系人 6.排序      *****|\n");printf("|*****7.清除所有联系人 0.退出程序   ****|\n");printf("|***************************************|\n");printf("————————————————————\n");
}
int main()
{int input = 0;//通讯录初始化contact pc;initCantact(&pc);do{menu();printf("请选择:\n");scanf("%d", &input);//switch (input){case 1:Addcontact(&pc);//增加联系人break;case 2:Delcontact(&pc);//删除联系人break;case 3:Modfycontatc(&pc);//修改联系人break;case 4:Searchcontact(&pc);//查找指定联系人break;case 5:Showcontact(&pc);//显示联系人break;case 6:Sortcontact(&pc);//按名字进行排序break;case 7://清空所以联系人Emptycontact(&pc);break;case 0:Destorycontacy(&pc);printf("退出程序\n");break;default:printf("输入错误,重新输入\n");}} while (input);return 0;
}

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

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

相关文章

《Linux从练气到飞升》No.15 Linux 环境变量

&#x1f57a;作者&#xff1a; 主页 我的专栏C语言从0到1探秘C数据结构从0到1探秘Linux菜鸟刷题集 &#x1f618;欢迎关注&#xff1a;&#x1f44d;点赞&#x1f64c;收藏✍️留言 &#x1f3c7;码字不易&#xff0c;你的&#x1f44d;点赞&#x1f64c;收藏❤️关注对我真的…

【linux基础(四)】对Linux权限的理解

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:Linux从入门到开通⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学更多操作系统知识   &#x1f51d;&#x1f51d; Linux权限 1. 前言2. shell命…

2023 最新 小丫软件库app开源源码 PHP后端

上传了源码解压之后&#xff0c;在admin/public/config.php修改后台登录账号和密码 后台地址&#xff1a;域名或者ip/admin 然后自己修改配置即可 后端搭建完成&#xff0c;现在导入iapp源码 导入iapp源码之后&#xff0c;修改mian.iyu载入事件的对接api和url就可以打包了 sss …

哈工大开源“活字”对话大模型

一、介绍 大规模语言模型&#xff08;LLM&#xff09;在自然语言处理的通用领域已取得了令人瞩目的成功。对于广泛的应用场景&#xff0c;这种技术展示了强大的潜力&#xff0c;学术界和工业界的兴趣也持续升温。哈工大自然语言处理研究所30余位老师和学生参与开发了通用对话大…

小米分享 | 解密面试题:网易面试如何回答“创建线程有哪几种方式?”

大家好&#xff0c;我是你们的小米&#xff01;今天要和大家一起探讨一个在技术面试中常见的问题&#xff1a;创建线程有哪几种方式&#xff1f;这可是个经典面试题哦&#xff01;不过别担心&#xff0c;小米在这里为你详细解析&#xff0c;帮你轻松应对&#xff0c;让你在面试…

stm32控制蜂鸣器源代码(附带proteus线路图)

说明&#xff1a; 1 PB0输出0时&#xff0c;蜂鸣器发生&#xff1b; 2 蜂鸣器电阻值如果太大会导致电流太小&#xff0c;发不出声音&#xff1b; 3蜂鸣器额定电压需要设置得低一点&#xff0c;可以是2V&#xff0c;但不能高于3V&#xff0c;这更右上角的电阻值有关系&#x…

【Linux】make/makefile自动化构建工具

文章目录 前言一、什么是make/makefile&#xff1f;二、依赖关系和依赖方法2.1 makefile中创建文件2.2 makefile中删除文件2.3 stat指令查看文件的三种时间&#xff08;ACM&#xff09;2.4 伪目标文件&#xff08;.PHONY&#xff09; 三、Makefile中的一些特殊符号3.1 $ 和 $^3…

【深入理解ES6】字符串和正则表达式

概念 字符串&#xff08;String&#xff09;是JavaScript6大原始数据类型。其他几个分别是Boolean、Null、Undefined、Number、Symbol&#xff08;es6新增&#xff09;。 更好的Unicode支持 1. UTF-16码位 字符串里的字符有两种&#xff1a; 前 个码位均以16位的编码单元…

尚硅谷大数据项目《在线教育之离线数仓》笔记002

视频地址&#xff1a;尚硅谷大数据项目《在线教育之离线数仓》_哔哩哔哩_bilibili 目录 P025 P026 P027 P028 P029 P030 P031 P032 P033 P034 P035 P036 P037 P038 P025 在Hive所在节点部署Spark P026 3&#xff09;Hive on Spark测试 &#xff08;1&#xff09;…

深度剖析数据在内存中的存储

目录 一、数据类型介绍 类型的基本归类 1.整形家族 2.浮点数家族 3.构造类型 &#xff08;自定义类型&#xff09; 4.指针类型 5.空类型 二、整形在内存中的存储 1.原码、反码、补码 1.1原码 1.2反码 1.3补码 1.4计算规则 2 .大小端介绍 三、浮点型在内存中的存…

C++函数模板和类模板

C另一种编程思想称为泛型编程&#xff0c;主要利用的技术是模板 C提供两种模板机制&#xff1a;函数模板和类模板 C提供了模板(template)编程的概念。所谓模板&#xff0c;实际上是建立一个通用函数或类&#xff0c; 其类内部的类型和函数的形参类型不具体指定&#xff0c; 用…

算法与数据结构(七)--堆

一.堆 1.堆的定义 堆是计算机科学中一类特殊的数据结构的通常&#xff0c;堆通常可以被看做是一颗完全二叉树的数组对象。 堆的特性 1.它是完全二叉树&#xff0c;除了树的最后一层结点不需要是满的&#xff0c;其他的每一层从左到右都是满的&#xff0c;如果最后一层结点不…

WordPress更换域名后-后台无法进入,网站模版错乱,css失效,网页中图片不显示。完整解决方案(含宝塔设置)

我在实际解决问题时用到了 【简单暴力解决方案】的《方法一&#xff1a;修改wp-config.php》 和 【简单暴力-且特别粗暴-的解决方案】 更换域名时经常遇到的几个问题&#xff1a; 1、更换域名后&#xff0c;后台无法进入 2、更换域名后&#xff0c;网站模版错乱&#xff0c;c…

基于物理场的动态模式分解(piDMD)研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

stm32红绿灯源代码示例(附带Proteus电路图)

本代码不能直接用于红路灯&#xff0c;只是提供一个思路 #include "main.h" #include "gpio.h" void SystemClock_Config(void); void MX_GPIO_Init(void) {GPIO_InitTypeDef GPIO_InitStruct {0};/* GPIO Ports Clock Enable */__HAL_RCC_GPIOB_CLK_ENAB…

九耶丨阁瑞钛伦特-在项目中找到的经典BUG是什么?

在项目中找到的经典BUG有很多种&#xff0c;以下是其中一些常见的例子&#xff1a; 空指针异常&#xff08;NullPointerException&#xff09;&#xff1a;当程序试图访问一个空对象或未初始化的变量时&#xff0c;会抛出空指针异常。这通常是由于缺少对变量的正确初始化或检查…

RabbitMq-发布确认高级(避坑指南版)

在初学rabbitMq的时候&#xff0c;伙伴们肯定已经接触到了“发布确认”的概念&#xff0c;但是到了后期学习中&#xff0c;会接触到“springboot”中使用“发布确认”高级的概念。后者主要是解决什么问题呢&#xff1f;或者是什么样的场景引出这样的概念呢&#xff1f; 在生产环…

第1天----验证一个字符串是否是另一个字符串的子串

本文我们将学习如何去验证一个字符串是否是另一个字符串的子串。 一、小试牛刀&#xff1a; 题目描述 输入两个字符串&#xff0c;验证其中一个串是否为另一个串的子串。 输入格式 两行&#xff0c;每行一个字符串。 输出格式 若第一个串 s 1 是第二个串 s 2 的子串&#xff0c…

[Machine Learning] decision tree 决策树

&#xff08;为了节约时间&#xff0c;后面关于机器学习和有关内容哦就是用中文进行书写了&#xff0c;如果有需要的话&#xff0c;我在目前手头项目交工以后&#xff0c;用英文重写一遍&#xff09; &#xff08;祝&#xff0c;本文同时用于比赛学习笔记和机器学习基础课程&a…

【java安全】Log4j反序列化漏洞

文章目录 【java安全】Log4j反序列化漏洞关于Apache Log4j漏洞成因CVE-2017-5645漏洞版本复现环境漏洞复现漏洞分析 CVE-2019-17571漏洞版本漏洞复现漏洞分析 参考 【java安全】Log4j反序列化漏洞 关于Apache Log4j Log4j是Apache的开源项目&#xff0c;可以实现对System.out…