[C语言]百行代码实现通讯录(进阶篇)

2024-01-24 03:04

本文主要是介绍[C语言]百行代码实现通讯录(进阶篇),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

前言:

1.进阶通讯录特点:

2.实现步骤:

(1)定义一个结构体来存储联系人的基本信息例如:名字、电话、性别等 ;

(2)定义另一个结构体来封装联系人及其个数以及记录容量;

(3)初始化通讯录;

(4)创建菜单栏;

(5)基本功能函数

(6)释放空间;

3.完整代码实现

(1)contact.h

(2)contact.c

(3)test.c

4.运行结果(如图)

5.结语


前言:

上篇我们通过结构体的构建来创造通讯录的联系人内容以及封装100个联系人,并实现了通讯录的增删查改等基本功能,这篇我们要进行改造的内容就是将固定的100个联系人容量改为不定长,有几个联系人就进行扩容,可以避免不必要的空间浪费。

1.进阶通讯录特点:

①基本的增删查改功能;

②通讯录的空间是不固定的,大小是可以调整

③默认能放3个人的信息,如果不够就每次增加2个人的信息

2.实现步骤:

(1)定义一个结构体来存储联系人的基本信息例如:名字、电话、性别等 ;

typedef struct PeoInfo 
{char name[MAX_NAME];char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];int age;
}PeoInfo;

(2)定义另一个结构体来封装联系人及其个数以及记录容量;

typedef struct Contact
{PeoInfo* data;//创建一个PeoInfo指针来指向存放数据的空间int sz;//用来记录存放联系人个数int capacity;//记录通讯录当前的最大容量
}Contact;

(3)初始化通讯录;

void InitContact(Contact* pc)
{assert(pc);//断言判断指针是否为空pc->data = malloc(DEFAULT_SZ * sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact");return;}pc->capacity = DEFAULT_SZ;pc->sz = 0;
}

(4)创建菜单栏;

void menu()
{printf("*******Contact******\n");printf("*** 1.ADD  2.DEL ***\n");printf("***** 3.SEARCH *****\n");printf("***** 4.MODIFY *****\n");printf("** 5.SHOE  6.SORT **\n");printf("****** 0.EXIT ******\n");}//后面通过case语句来进行你想要的操作

(5)基本功能函数

可以通过http://t.csdnimg.cn/gIo96百行代码实现简单通讯录来查看哦;这里将增加联系人函数单独拿出来,因为有改进的地方

int CheckCapacity(Contact* pc)//增容函数 加联系人时如果空间不够就扩容
{if (pc->sz == pc->capacity){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INT_SZ)*sizeof(PeoInfo));if (ptr == NULL){perror("CheckCapacity");return 0;}else{pc->data = ptr;pc->capacity += INT_SZ;printf("增容成功\n");}return 1;}
}void AddContact(Contact* pc)//增加联系人
{assert(pc);if (0 == CheckCapacity(pc)){printf("无法增容哦~");return;}printf("请输入要添加的联系人名字、性别、年龄、电话、地址:\n");scanf("%s%s%d%s%s",pc->data[pc->sz].name, pc->data[pc->sz].sex,&pc->data[pc->sz].age, pc->data[pc->sz].tele,pc->data[pc->sz].addr);printf("您已经成功添加%s\n", pc->data[pc->sz].name);pc->sz++;return;
}

(6)释放空间;

使用了malloc,realloc函数进行开辟空间并增容一定记得释放空间并将指针置空

void DestroyContact(Contact* pc)//释放空间
{free(pc->data);pc->data = NULL;pc->capacity = 0;pc->sz = 0;
}

3.完整代码实现

(1)contact.h

#pragma once
//定义的的头文件contact.h
#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.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 INT_SZ 2
enum OPTION//用枚举来定义变量
{EXIT,//0ADD,//1DEL,//2...SEARCH,MODIFY,SHOW,SORT
};//声明类型
typedef struct PeoInfo //创建一个结构体来储存联系人相关信息
{char name[MAX_NAME];char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];int age;
}PeoInfo;//通讯录
typedef struct Contact
{PeoInfo* data;//创建一个PeoInfo指针来指向存放数据的空间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 DestroyContact(Contact* pc);//释放空间

(2)contact.c

#define _CRT_SECURE_NO_WARNINGS 1
//contact函数实现contact.c源文件
#define _CRT_SECURE_NO_WARNINGS 1
#include"contact.h"
int Search_by_name(const Contact* pc)//查找函数
{char name[MAX_NAME] = { 0 };scanf("%s", name);int i = 0;for (i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, name) == 0){return i;}}return 0;
}//静态版本
//void InitContact(Contact* pc) //初始化通讯录函数
//{
//	memset(pc->data, 0, sizeof(pc->data));
//	pc->sz = 0;
//}
//
//动态版本
void InitContact(Contact* pc)
{assert(pc);//断言判断指针是否为空pc->data = malloc(DEFAULT_SZ * sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact");return;}pc->capacity = DEFAULT_SZ;pc->sz = 0;
}int CheckCapacity(Contact* pc)
{if (pc->sz == pc->capacity){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INT_SZ)*sizeof(PeoInfo));if (ptr == NULL){perror("CheckCapacity");return 0;}else{pc->data = ptr;pc->capacity += INT_SZ;printf("增容成功\n");}return 1;}
}void AddContact(Contact* pc)//增加联系人
{assert(pc);if (0 == CheckCapacity(pc)){printf("无法增容哦~");return;}printf("请输入要添加的联系人名字、性别、年龄、电话、地址:\n");scanf("%s%s%d%s%s",pc->data[pc->sz].name, pc->data[pc->sz].sex,&pc->data[pc->sz].age, pc->data[pc->sz].tele,pc->data[pc->sz].addr);printf("您已经成功添加%s\n", pc->data[pc->sz].name);pc->sz++;return;
}void ShowContact(const Contact* pc)//显示联系人
{assert(pc);if (pc->sz == 0){printf("您还未添加联系人哦~快选择1去添加吧~\n");return;}int i = 0;printf("%-10s\t%-5s\t%-5s\t%-15s\t%-30s\t\n", "名字", "性别", "年龄", "电话", "住址");for (i = 0; i < pc->sz; i++){printf("%-10s\t%-5s\t%-5d\t%-15s\t%-30s\t\n",pc->data[i].name, pc->data[i].sex,pc->data[i].age, pc->data[i].tele,pc->data[i].addr);}return;
}void DelContact(Contact* pc)//删除联系人
{assert(pc);if (pc->sz == 0){printf("您还没有加联系人哦~快选择1去添加吧~\n");return;}printf("请输入要删除的联系人的名字:\n");int flag = Search_by_name(&pc);if (flag == 0){printf("没有找到该联系人哦~\n");return;}int j = 0;for (j = flag; j < pc->sz - 1; j++){pc->data[j] = pc->data[j + 1];if (strcmp(pc->data[j + 1].name, 0) == 0){break;}}printf("您已经成功删除该联系人\n");pc->sz--;return;
}void SearchContact(const Contact* pc)//查找联系人并打印
{int i = Search_by_name(&pc);if (i == 0){printf("没有找到该联系人哦~\n");return;}else{printf("%-10s\t%-5s\t%-5d\t%-15s\t%-30s\t\n",pc->data[i].name, pc->data[i].sex,pc->data[i].age, pc->data[i].tele,pc->data[i].addr);}return;
}void DestroyContact(Contact* pc)//释放空间
{free(pc->data);pc->data = NULL;pc->capacity = 0;pc->sz = 0;
}

(3)test.c

#define _CRT_SECURE_NO_WARNINGS 1
//test.c文件--流程
#define _CRT_SECURE_NO_WARNINGS 1
#include"contact.h"
void menu()//菜单
{printf("*******Contact******\n");printf("*** 1.ADD  2.DEL ***\n");printf("***** 3.SEARCH *****\n");printf("***** 4.MODIFY *****\n");printf("** 5.SHOE  6.SORT **\n");printf("****** 0.EXIT ******\n");}
int main()//主函数
{int sec = 1;Contact con;InitContact(&con);//初始化通讯录while (sec){menu();scanf("%d", &sec);switch (sec)//选择不同的功能{case EXIT://退出DestroyContact(&con);printf("您已退出\n");break;case ADD://加AddContact(&con);break;case DEL://删DelContact(&con);break;case SEARCH://查SearchContact(&con);break;case MODIFY://改break;case SHOW://显示ShowContact(&con);break;case SORT://分类break;default:printf("选择错误,请重新输入\n");}printf("\n");}return 0;
}

4.运行结果(如图)

5.结语

以上就是通讯录不定容实现增删查改功能的完整代码啦,结合了结构体以及malloc,realloc相关内存函数来实现。 

这篇关于[C语言]百行代码实现通讯录(进阶篇)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Python FastAPI+Celery+RabbitMQ实现分布式图片水印处理系统

《PythonFastAPI+Celery+RabbitMQ实现分布式图片水印处理系统》这篇文章主要为大家详细介绍了PythonFastAPI如何结合Celery以及RabbitMQ实现简单的分布式... 实现思路FastAPI 服务器Celery 任务队列RabbitMQ 作为消息代理定时任务处理完整

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

使用C#代码在PDF文档中添加、删除和替换图片

《使用C#代码在PDF文档中添加、删除和替换图片》在当今数字化文档处理场景中,动态操作PDF文档中的图像已成为企业级应用开发的核心需求之一,本文将介绍如何在.NET平台使用C#代码在PDF文档中添加、... 目录引言用C#添加图片到PDF文档用C#删除PDF文档中的图片用C#替换PDF文档中的图片引言在当

C#使用SQLite进行大数据量高效处理的代码示例

《C#使用SQLite进行大数据量高效处理的代码示例》在软件开发中,高效处理大数据量是一个常见且具有挑战性的任务,SQLite因其零配置、嵌入式、跨平台的特性,成为许多开发者的首选数据库,本文将深入探... 目录前言准备工作数据实体核心技术批量插入:从乌龟到猎豹的蜕变分页查询:加载百万数据异步处理:拒绝界面

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("