【C语言】通讯录2.0 (动态增长版)

前言

通讯录是一种记录联系人信息的工具,包括姓名、电话号码、电子邮件地址、住址等。
文章的一二三章均于上一篇相同,可以直接看第四章改造内容。
此通讯录是基于通讯录1.0(静态版)的基础上进行改进,请先看系列文字第一篇,再看本篇博客。
****** 有需要源代码,见文章末尾 ******


系列文章目录

第一篇:【C语言】通讯录1.0 (静态版)
第二篇:【C语言】通讯录2.0 (动态增长版)
第三篇:【C语言】通讯录2.0 (文件存储版)


文章目录

  • 前言
  • 系列文章目录
  • 一、什么是通讯录
  • 二、静态版、动态增长版和文件存储版的区别
    • 1. 静态版
    • 2. 动态增长版
    • 3. 文件存储版
  • 三、通讯录模块组成(图文)
    • 1. 通讯录文件构成
    • 2. 通讯录个人信息
    • 3. 通讯录功能模块
  • 四、如何改造通讯录1.0(改造目标)
    • 1. 改造目标
    • 2. 需要的改造部分
  • 五、如何改进(代码演示)
    • 1. 通讯录结构模块
    • 2. 通讯录初始化函数
    • 3. 增加联系人函数
    • 4. 添加内存释放函数
  • 六、所有文件代码
    • 1. 头文件
    • 2. 函数文件
    • 3. 测试逻辑文件
  • 总结


一、什么是通讯录

通讯录是一种记录联系人信息的工具,包括姓名、电话号码、电子邮件地址、住址等。通讯录可以帮助人们管理自己的联系人,让人们更轻松地与他人保持联系。通讯录可以在手机、电脑、笔记本等设备上保存,也可以在云端储存和同步,方便用户随时查看和更新联系人信息。

二、静态版、动态增长版和文件存储版的区别

C语言静态版、动态增长版和文件存储版的区别如下:

1. 静态版

  1. 静态版:在程序编译时就确定了内存大小,程序运行期间内存大小不会发生变化,因此对于需要处理大量数据或者不确定数据大小的情况不适用。

2. 动态增长版

  1. 动态增长版:可以在程序运行期间根据需要动态增加内存大小,因此适用于处理不确定数据大小的情况。但是动态增长的内存需要手动释放,否则会导致内存泄漏。

3. 文件存储版

  1. 文件存储版:将数据存储在文件中,可以持久保存数据并随时读取。但是存储在文件中的数据需要进行IO操作,因此相比于内存操作来说效率较低。此外,文件存储版不适用于需要频繁修改的数据。

三、通讯录模块组成(图文)

1. 通讯录文件构成

在这里插入图片描述

2. 通讯录个人信息

在这里插入图片描述

3. 通讯录功能模块

在这里插入图片描述

四、如何改造通讯录1.0(改造目标)

1. 改造目标

  1. 通讯录的空间不是固定的,大小是可以调整的
  2. 默认能放3个人的信息,如果不够,就每次增加2个人的信息

2. 需要的改造部分

在这里插入图片描述

五、如何改进(代码演示)

1. 通讯录结构模块

  • 将通讯录的存储方式改成动态增长,用指针来调用
  • 需要增加通讯录的现有的容量值
//动态版
typedef struct Contact
{PeoInfo* data;int sz;int capacity;}Contact;

2. 通讯录初始化函数

  • 本来初始化通讯录是使用内存函数memset来实现
  • 现在由malloc动态内存函数来动态开辟内存
  • 初始化设置容量为3个联系人的容量
//动态版
void InitContact(Contact* pc)
{assert(pc);pc->data = (PeoInfo*)malloc(3 * sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact");return;}pc->capacity = 3;pc->sz = 0;
}

3. 增加联系人函数

  • 本来前提是判断是否通讯录已满,现在改为判断当通讯录容量满的时候增加动态内存
  • 需要写一个函数来判断容量是否已满
  • 当容量已满的时候。使用realloc动态内存函数,来增加动态内存
//动态版
void AddContact(Contact* pc)
{assert(pc);       //断言if (determine(pc) == 0){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++;    //通讯录加1printf("联系人增加成功\n");
}
//判断内存是否已满函数
int determine(Contact* pc)
{assert(pc);if (pc->capacity == pc->sz){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));if (ptr == NULL){perror("determine");return 0;}else{pc->capacity += 2;pc->data = ptr;printf("增容成功\n");return 1;}}return 1;
}

4. 添加内存释放函数

  • 在通讯录菜单栏,选择退出 0 通讯录时要释放内存
  • 也要将容量和size 归零
  • 需要写一个函数来进行这个操作
void DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->capacity = 0;pc->sz;
}

六、所有文件代码

1. 头文件

#pragma once
#include<stdio.h>
#include<assert.h>
#include<string.h>
#include<stdlib.h>#define MAX 100
#define NAME 10
#define SEX  5
#define TELE 12
#define ADDR 30
//使用枚举  定义选择   
enum OPTION
{EXIT,//0ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};//个人信息类型声明
typedef struct PeoInfo
{char name[NAME];int age;char sex[SEX];char tele[TELE];char addr[ADDR];
}PeoInfo;//建立通讯录
//静态版
//typedef struct Contact
//{
//	PeoInfo data[MAX]; //通讯录数量
//	int sz;				//目前通讯录内的人数	
//}Contact;
//动态版
typedef struct Contact
{PeoInfo* data;int sz;int capacity;}Contact;//函数声明//初始化通讯录
void InitContact(Contact* pc);//增加联系人
void AddContact(Contact* pc);//显示所有联系人的信息
void ShowContact(const Contact* pc);//删除指定联系人
void DelContact(Contact* pc);//查找指定联系人
void SearchContact(const Contact* pc);//修改指定联系人
void ModifyContact(Contact* pc);//释放内存void DestroyContact(Contact* pc);

2. 函数文件

#define _CRT_SECURE_NO_WARNINGS 1
#include "addbook.h"int determine(Contact* pc)
{assert(pc);if (pc->capacity == pc->sz){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));if (ptr == NULL){perror("determine");return 0;}else{pc->capacity += 2;pc->data = ptr;printf("增容成功\n");return 1;}}return 1;
}
//静态版
//void InitContact(Contact* pc)
//{
//	assert(pc);   //断言
//	memset(pc->data, 0, sizeof(pc->data));   //内存函数  data初始化为0  
//	pc->sz = 0;
//}
//动态版
void InitContact(Contact* pc)
{assert(pc);pc->data = (PeoInfo*)malloc(3 * sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact");return;}pc->capacity = 3;pc->sz = 0;
}
//动态版
void AddContact(Contact* pc)
{assert(pc);       //断言if (determine(pc) == 0){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++;    //通讯录加1printf("联系人增加成功\n");
}//静态版
//void AddContact(Contact* pc)
//{
//	assert(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);
//	pc->sz++;    //通讯录加1
//	printf("联系人增加成功\n");
//}
//搜索名字找通讯录函数
static int Findname(const Contact* pc, char na[])
{int i = 0;assert(pc && na);for (i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, na) == 0){return i;}}return -1;
}
void DelContact(Contact* pc)
{if (pc->sz == 0){printf("通讯录为空\n");return;}char name[NAME] = { 0 };assert(pc);//输入要查找的联系人名字printf("请输入要查找的名字:>");scanf("%s", &name);//找到要查找的联系人int del = Findname(pc, name);//删除坐标位子的联系人 ,将后面的联系人进行代替其位置if (del == -1){printf("找不到,此人不存在\n");return;}else{int i = 0;for (i = del; i < pc->sz; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;}printf("成功删除联系人\n");
}
void ShowContact(const Contact* pc)
{assert(pc);printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");int i = 0;for (i = 0; i < pc->sz; i++){printf("%-10s\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);}printf("通讯录展示完毕\n");
}
void SearchContact(const Contact* pc)
{if (pc->sz == 0){printf("通讯录为空\n");return;}char name[NAME] = { 0 };assert(pc);//输入要查找的联系人名字printf("请输入要查找的名字:>");scanf("%s", &name);//找到要查找的联系人int i = Findname(pc, name);if (i == -1){printf("找不到,此人不存在\n");return;}else{printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("%-10s\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);}printf("成功找到联系人\n");
}
void ModifyContact(Contact* pc)
{assert(pc);char name[NAME] = { 0 };printf("请输入要修改人的名字:>");scanf("%s", &name);int mod = Findname(pc, name);if (mod == -1){printf("找不到,不存在\n");return;}else{printf("请输入姓名:>");scanf("%s", pc->data[mod].name);printf("请输入年龄:>");scanf("%d", &(pc->data[mod].age));printf("请输入性别:>");scanf("%s", pc->data[mod].sex);printf("请输入电话:>");scanf("%s", pc->data[mod].tele);printf("请输入地址:>");scanf("%s", pc->data[mod].addr);printf("联系人修改成功\n");}
}
void DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->capacity = 0;pc->sz;
}

3. 测试逻辑文件

#define _CRT_SECURE_NO_WARNINGS 1
#include"addbook.h"
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");}void test()
{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:printf("功能待开发\n");break;case EXIT:DestroyContact(&con);printf("成功退出通讯录\n");break;default:printf("输入错误,请重新输入\n");break;}} while (input);}
int main()
{test();return 0;
}

总结

本期博客,是通讯录2.0(动态增长版),是对前面所学知识进行复习,编写通讯录时有助于理解自定义类型和动态内存管理的学习和了解,后期会对现在的通讯录进行更新!!!


如这篇博客对大家有帮助的话,希望 三连 支持一下 !!! 如果有错误感谢大佬的斧正 如有 其他见解发到评论区,一起学习 一起进步。

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

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

相关文章

自动化测试:让软件测试更高效更愉快!

谈谈那些实习测试工程师应该掌握的基础知识&#xff08;一&#xff09;_什么时候才能变强的博客-CSDN博客https://blog.csdn.net/qq_17496235/article/details/131839453谈谈那些实习测试工程师应该掌握的基础知识&#xff08;二&#xff09;_什么时候才能变强的博客-CSDN博客h…

css 动画之旋转视差

序&#xff1a;网上看到的一个例子&#xff0c;做一下 效果图&#xff1a; 代码&#xff1a; <style>.content{width: 300px;height: 300px;margin: 139px auto;display: grid;grid-template-columns: repeat(3,1fr);grid-template-rows: repeat(3,1fr);grid-template:…

Python 进阶(六):文件读写(I/O)

❤️ 博客主页&#xff1a;水滴技术 &#x1f338; 订阅专栏&#xff1a;Python 入门核心技术 &#x1f680; 支持水滴&#xff1a;点赞&#x1f44d; 收藏⭐ 留言&#x1f4ac; 文章目录 1. 打开文件2. 读取文件2.1 逐行读取文件2.2 读取所有行 3. 写入文件3.1 向文件中写入…

从0到1开发go-tcp框架【1-搭建server、封装连接与业务绑定、实现基础Router、抽取全局配置文件】

从0到1开发go-tcp框架【1-搭建server、封装连接与业务绑定、实现基础Router】 本期主要完成对Server的搭建、封装连接与业务绑定、实现基础Router&#xff08;处理业务的部分&#xff09;、抽取框架的全局配置文件 从配置文件中读取数据&#xff08;服务器监听端口、监听IP等&a…

汇编语言基础知识

目录 前言&#xff1a; 汇编语言的产生 汇编语言的组成 内存 指令和数据 cpu对内存的读写操作 地址总线 数据总线 控制总线 内存地址空间 前言&#xff1a; 汇编语言是直接在硬件之上工作的 编程语言&#xff0c;我们首先了解硬件系统的机构&#xff0c;才能有效地应用…

C/C++多线程操作

文章目录 多线程C创建线程join 和detachthis_thread线程操作锁lock_guardunique_lock 条件变量 condition_variablewaitwaitfor C语言线程创建线程同步 参考 多线程 传统的C&#xff08;C11标准之前&#xff09;中并没有引入线程这个概念&#xff0c;在C11出来之前&#xff0c…

【编译】gcc make cmake Makefile CMakeList.txt 区别

文章目录 一 关系二 gcc2.1 编译过程2.2 编译参数2.3 静态库和动态库1 后缀名2 联系与区别 2.4 GDB 调试器1 常用命令 三 make、makefile四 cmake、cmakelist4.1 语法特性4.2 重要命令4.2 重要变量4.3 编译流程4.4 两种构建方式 五 Vscode5.0 常用快捷键5.1 界面5.2 插件5.3 .v…

STM32 SPI学习

SPI 串行外设设备接口&#xff08;Serial Peripheral Interface&#xff09;&#xff0c;是一种高速的&#xff0c;全双工&#xff0c;同步的通信总线。 SCK时钟信号由主机发出。 SPI接口主要应用在存储芯片。 SPI相关引脚&#xff1a;MOSI&#xff08;输出数据线&#xff…

1.netty介绍

1.介绍 是JBOSS通过的java开源框架是异步的,基于事件驱动(点击一个按钮调用某个函数)的网络应用框架,高性能高可靠的网络IO程序基于TCP,面向客户端高并发应用/点对点大量数据持续传输的应用是NIO框架 (IO的一层层封装) TCP/IP->javaIO和网络编程–>NIO—>Netty 2.应用…

性能测试必备监控技能windows篇

前言 在手头没有专门的第三方监控时&#xff0c;该怎么监控服务指标呢&#xff1f;本篇就windows下监控进行分享&#xff0c;也是我们在进行性能测试时&#xff0c;必须掌握的。下面我们就windows下常用的三种监视工具进行说明&#xff1a; 任务管理器 资源监视器 性能监视器…

找样机素材,就上这5个网站,免费下载~

设计师经常需要用到各种样机模型来展示直接的作品&#xff0c;今天我就分享几个可以免费下载样机模型的网站&#xff0c;大家赶紧收藏起来&#xff01; 菜鸟图库 https://www.sucai999.com/searchlist/3217.html?vNTYxMjky 菜鸟图库有多种类型的设计素材&#xff0c;像平面、…

Element-plus侧边栏踩坑

问题描述 el-menu直接嵌套el-menu-item菜单&#xff0c;折叠时不会出现文字显示和小箭头无法隐藏的问题&#xff0c;但是实际开发需求中难免需要把el-menu-item封装为组件 解决 vue3项目中嵌套两层template <template><template v-for"item in list" :k…

linux V4L2子系统——v4l2架构(1)之整体架构

概述 V4L&#xff08;Video for Linux&#xff09;是Linux内核中关于视频设备的API接口&#xff0c;涉及视频设备的音频和视频信息采集及处理、视频设备的控制。V4L出现于Linux内核2.1版本&#xff0c;经过修改bug和添加功能&#xff0c;Linux内核2.5版本推出了V4L2&#xff08…

MySQL主从复制及读写分离(三十四)

目录 MySQL主从复制 一、概述 1、MySQL Replication优点&#xff1a; 二、MySQL复制类型 1、异步复制&#xff08;Asynchronous repication&#xff09; 2、全同步复制&#xff08;Fully synchronous replication&#xff09; 3、半同步复制&#xff08;Semisynchronous…

短视频矩阵源码开发搭建分享--多账号授权管理

目录 文章目录 前言 一、矩阵号系统是什么&#xff1f; 二、使用步骤 1.创建推广项目 2.多账号授权 3.企业号智能客服系统 总结 前言 短视频多账号矩阵系统&#xff0c;通过多账号一键授权管理的方式&#xff0c;为运营人员打造功能强大及全面的“矩阵式“管理平台。…

【shell】获取ping的时延数据并分析网络情况

网络情况经常让我们头疼&#xff0c;每次都需要手动在终端ping太麻烦了&#xff0c;不如写个脚本ping并将数据带上时间戳存入文件&#xff0c;然后也可以分析哪个时间段网络比较差。 创建一个demo.sh文件&#xff1a; #!/bin/bash # 清理日志 net_path"./network/"…

STM32 USB使用记录:HID类设备(后篇)

文章目录 目的基础说明项目构建与代码调整接收发送代码与测试示例链接报告描述符总结 目的 接上篇&#xff1a; 《STM32 USB使用记录&#xff1a;HID类设备&#xff08;前篇&#xff09;》 USB HID 类的设备有个比较大的好处是大部分时候接入主机中都是可以免驱使用的。这篇文…

高并发架构去重难?架构必备技能 - 布隆过滤器

系列文章目录 当Dubbo遇到高并发&#xff1a;探究流量控制解决方案 主从选举机制&#xff0c;架构高可用性的不二选择 高并发架构去重难&#xff1f;架构必备技能 - 布隆过滤器 系列文章目录前言一、布隆过滤器简介二、特性与应用场景三、参数定制四、java版本的Demo五、总结 …

<findbugs>静态代码分析工具

背景&#xff1a; IDEA安装的findbug插件目前无法和jenkins的扫描结果保持一致&#xff0c;因为&#xff1a;没有对应jenkins上findbug的版本&#xff1b; 原理&#xff1a; 将jenkins服务器上的findbugs插件&#xff0c;拷贝到本地&#xff0c;修改build.xml内容以匹配目录…

Resnet与Pytorch花图像分类

1、介绍 1.1数据集介绍 flower_data├── train│ └── 1-102&#xff08;102个文件夹&#xff09;│ └── XXX.jpg&#xff08;每个文件夹含若干张图像&#xff09;├── valid│ └── 1-102&#xff08;102个文件夹&#xff09;└── ─── └── XXX.jp…