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

news2025/1/11 5:44:03

目录

前言:

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,//0
	ADD,//1
	DEL,//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相关内存函数来实现。 

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

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

相关文章

C语言零基础入门第1天《什么是C语言?》

C语言零基础入门第1天《什么是C语言&#xff1f;开篇点题》 1、什么是C语言2、为什么要使用 C&#xff1f;3、C有哪些好处、优点&#xff1f; 4、C排行榜排名5、C语言为什么是编程语言的神&#xff1f;6、鼓励你&#xff0c;为你叫好&#xff01; C语言——编程之本&#xff0c…

vulhub之Zabbix篇

CVE-2016-10134--SQL注入 一、漏洞介绍 zabbix是一款服务器监控软件&#xff0c;其由server、agent、web等模块组成&#xff0c;其中web模块由PHP编写&#xff0c;用来显示数据库中的结果。 漏洞环境 在vulhub靶场进行复现&#xff0c;启动zabbix 3.0.3。 二、复现步骤 1…

L1-096 谁管谁叫爹(Java)

《咱俩谁管谁叫爹》是网上一首搞笑饶舌歌曲&#xff0c;来源于东北酒桌上的助兴游戏。现在我们把这个游戏的难度拔高一点&#xff0c;多耗一些智商。 不妨设游戏中的两个人为 A 和 B。游戏开始后&#xff0c;两人同时报出两个整数 NA 和 NB。判断谁是爹的标准如下&#xff1a; …

一文让你彻底搞懂cookie和session产生漏洞的原理

首先让我们来看看登录的一般流程&#xff1a; 输入账号密码提交给后端&#xff1b;后端进行判断账号密码是否一致&#xff0c;这里的逻辑根据每个程序员的想法去写&#xff1b;如果通过2登录成功&#xff0c;跳转登录成功的页面&#xff1b; 如果通过2登录失败&#xff0c;跳转…

每日一题——LeetCode1309.解码字母到整数映射

方法一 个人方法&#xff1a; 由题意可得&#xff1a; 1、1-26对应a-z&#xff0c;只不过从10开始&#xff0c;后面要加#才能表示字母 2、10# - 26# 都是三位数&#xff0c;开头<2&#xff0c;结尾都为# 3、某个数后面两位如果不为#那么该数为1-9对应a-i&#xff0c;如果…

pikachu_csrf通关攻略

csrf&#xff08;get&#xff09; 打开pikachu靶场&#xff1a; 1. 根据提示给的账户密码进行登录 2. 打开代理拦截数据包将拦截数据发送到已打开的burp中&#xff1a; 修改数据进行发包&#xff1a; 从上面的url可见&#xff0c;修改用户信息的时候&#xff0c;是不带任何不…

npm ERR! code CERT_HAS_EXPIRED npm ERR! errno CERT_HAS_EXPIRED

npm install时报错code CERT_HAS_EXPIRED 一、报错情况二、解决方案 一、报错情况 一直用的好好的&#xff0c;突然今天发现npm install 出问题了&#xff0c;具体报错如下&#xff1a; npm ERR! code CERT_HAS_EXPIRED npm ERR! errno CERT_HAS_EXPIRED npm ERR! request to…

Flink中的时间和窗口(时间语义,水位线,窗口,迟到数据的处理)

目录 Flink中的时间和窗口 1时间语义 1.1Flink中的时间语义 1.1.1处理时间 1.1.2事件时间 1.2那种时间语义更重要 2 水位线 2.1 事件时间和窗口 2.2 什么是水位线 2.3 如何生成水位线 2.3.1使用WatermarkGenerator 2.3.2使用SourceFunction 2.4 水位线的传递 2.5 水位…

【LeetCode每日一题】670. 最大交换

2024-1-22 文章目录 [670. 最大交换](https://leetcode.cn/problems/maximum-swap/)思路&#xff1a; 670. 最大交换 思路&#xff1a; 从后往前遍历整数的字符数组&#xff0c;找到第一个比当前最大值小的数字&#xff0c;然后将该数字与最大值进行交换。 将给定的整数转换为…

PALWORLD linux centos7开服教程

开放端口8211 iptables -A INPUT -p udp -p tcp --dport 8211 -j ACCEPT iptables -A INPUT -p udp -p udp --dport 8211 -j ACCEPT yum -y install glibc.i686 libstdc.i686 screen libcurl.i686 yum install glibc-devel.i686 yum install libstdc-devel.i686 yum -y ins…

若依框架教程

心血来潮&#xff0c;写一篇关于ruoyi【若依】框架从0-1的教程。说一下使用感受吧&#xff0c;如果有一个架构师或者老手已经把架构改造完成也能指导你如何快速上手&#xff0c;那么你在后面的增删改查系列开发起来会如鱼得水。如果没有人改造&#xff0c;也没有人教你&#xf…

vue3源码(一)搭建开发环境

Monorepo:是管理项目代码的一个方式&#xff0c;指在一个项目仓库(repo)中管理多个模块/包(package) vue3源码采用Monorepo管理项目 vue3项目架构 搭建Monorepo环境 1.全局安装pnpm npm install pnpm -g # 全局安装pnpm pnpm init -y # 初始化配置文件创建.npmrc文件&#x…

小红书多模态团队建立新「扩散模型」:解码脑电波,高清还原人眼所见

近些年&#xff0c;研究人员们对探索大脑如何解读视觉信息&#xff0c;并试图还原出原始图像一直孜孜不倦。去年一篇被 CVPR 录用的论文&#xff0c;通过扩散模型重建视觉影像&#xff0c;给出了非常炸裂的效果—— AI 不光通过脑电波知道你看到了什么&#xff0c;并且帮你画了…

强化学习13——Actor-Critic算法

Actor-Critic算法结合了策略梯度和值函数的优点&#xff0c;我们将其分为两部分&#xff0c;Actor&#xff08;策略网络&#xff09;和Critic&#xff08;价值网络&#xff09; Actor与环境交互&#xff0c;在Critic价值函数的指导下使用策略梯度学习好的策略Critic通过Actor与…

【BBuf的CUDA笔记】十三,OpenAI Triton 入门笔记一

0x0. 前言 2023年很多mlsys工作都是基于Triton来完成或者提供了Triton实现版本&#xff0c;比如现在令人熟知的FlashAttention&#xff0c;大模型推理框架lightllm&#xff0c;diffusion第三方加速库stable-fast等灯&#xff0c;以及很多mlsys的paper也开始使用Triton来实现比…

瑞_力扣LeetCode_104. 二叉树的最大深度

文章目录 题目 104. 二叉树的最大深度题解后序遍历 递归实现后序遍历 迭代实现层序遍历 题目 111. 二叉树的最小深度题解后序遍历层序遍历 题目 226. 翻转二叉树题解 &#x1f64a; 前言&#xff1a;本文章为瑞_系列专栏之《刷题》的力扣LeetCode系列&#xff0c;主要以力扣Lee…

Pandas.Series.sum() 求和(累和) 详解 含代码 含测试数据集 随Pandas版本持续更新

关于Pandas版本&#xff1a; 本文基于 pandas2.2.0 编写。 关于本文内容更新&#xff1a; 随着pandas的stable版本更迭&#xff0c;本文持续更新&#xff0c;不断完善补充。 传送门&#xff1a; Pandas API参考目录 传送门&#xff1a; Pandas 版本更新及新特性 传送门&…

sqlmap使用教程(3)-探测注入漏洞

1、探测GET参数 以下为探测DVWA靶场low级别的sql注入&#xff0c;以下提交方式为GET&#xff0c;问号&#xff08;?&#xff09;将分隔URL和传输的数据&#xff0c;而参数之间以&相连。--auth-credadmin:password --auth-typebasic &#xff08;DVWA靶场需要登录&#xf…

【GitHub项目推荐--微软开源的课程(Web开发课程/机器学习课程/物联网课程/数据科学课程)】【转载】

微软在 GitHub 开源了四大课程&#xff0c;面向计算机专业或者入门编程的同学。分别是 Web 开发课程、机器学习课程、物联网课程和数据分析课程。 四大课程在 GitHub 上共斩获 90K 的Star&#xff0c;每一课程包含 20 多小节&#xff0c;完成课程大约需要 12 周。每小节除了视…

《统计学习方法:李航》笔记 从原理到实现(基于python)-- 第1章 统计学习方法概论

文章目录 第1章 统计学习方法概论1.1 统计学习1&#xff0e;统计学习的特点2&#xff0e;统计学习的对象3&#xff0e;统计学习的目的4&#xff0e;统计学习的方法1.2.1 基本概念1.2.2 问题的形式化 1.3 统计学习三要素1.3.1 模型1.3.2 策略1.3.3 算法 1.4 模型评估与模型选择1…