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

这篇具有很好参考价值的文章主要介绍了【C语言】通讯录2.0 (动态增长版)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

前言

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


系列文章目录

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



一、什么是通讯录

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

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

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

1. 静态版

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

2. 动态增长版

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

3. 文件存储版

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

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

1. 通讯录文件构成

【C语言】通讯录2.0 (动态增长版),C语言,数据结构,c语言,开发语言,数据结构,c++

2. 通讯录个人信息

【C语言】通讯录2.0 (动态增长版),C语言,数据结构,c语言,开发语言,数据结构,c++

3. 通讯录功能模块

【C语言】通讯录2.0 (动态增长版),C语言,数据结构,c语言,开发语言,数据结构,c++

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

1. 改造目标

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

2. 需要的改造部分

【C语言】通讯录2.0 (动态增长版),C语言,数据结构,c语言,开发语言,数据结构,c++

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

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++;    //通讯录加1
	printf("联系人增加成功\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,//0
	ADD,
	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++;    //通讯录加1
	printf("联系人增加成功\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(动态增长版),是对前面所学知识进行复习,编写通讯录时有助于理解自定义类型和动态内存管理的学习和了解,后期会对现在的通讯录进行更新!!!


如这篇博客对大家有帮助的话,希望 三连 支持一下 !!! 如果有错误感谢大佬的斧正 如有 其他见解发到评论区,一起学习 一起进步。文章来源地址https://www.toymoban.com/news/detail-620193.html

到了这里,关于【C语言】通讯录2.0 (动态增长版)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C语言】动态通讯录 -- 详解

    前面详细介绍了静态版通讯录【C语言】静态通讯录 -- 详解_炫酷的伊莉娜的博客-CSDN博客,但是静态版通讯录的空间是无法被改变的,而且空间利用率也不高。为了解决静态通讯录这一缺点,这时就要有一个能够随着存入联系人数量的增加而增大空间的通讯录。接下来我们将

    2024年02月12日
    浏览(41)
  • 动态通讯录实现(C语言)

    目录 前言: 一:单个节点的设计和主逻辑  结点设计 主逻辑 二:接口实现 (1)生成一个新的结点 (2)增加信息 (3)打印信息 (4)查找  (5)删除信息 (6)修改信息 (7)排序  插入排序 快速排序 (8)已有数据读取 (9)更新数据录入 三:全部代码 contact.h(声明) contact.c(接口) test.c(主逻辑) 本

    2024年02月05日
    浏览(54)
  • 【C语言】动态通讯录(超详细)

    通讯录是一个可以很好锻炼我们对结构体的使用,加深对结构体的理解,在为以后学习数据结构打下结实的基础 这里我们想设计一个有 添加联系人,删除联系人,查找联系人,修改联系人,展示联系人,排序 这几种功能的通讯录 注意:我们按照三个区域划分 上图所示进行

    2024年02月08日
    浏览(34)
  • 【C语言】实现通讯录(动态+文件)

    在之前三子棋和扫雷的基础上,本篇文章博主将给大家逐步分析实现通讯录,介绍通讯录的每个功能( 动态增长和文件保存 )。 —————————————————————— test.c - 测试通讯录 Contact.c - 函数的实现 Contact.h - 函数和类型的声明 以多文件的形式分模块写的

    2024年02月13日
    浏览(54)
  • C语言实现通讯录--动态版

    实现一个通讯录,联系人的数量可多可少 1.在静态版本的基础上改用动态的方法: (1)默认能够存放三个人的信息 (2)不够的话,每次增加两个人的信息 2.其他功能不变 建立三个文件: test.c 用于测试通讯录的相关功能 contsct.c 通讯录的实现模块(用函数实现功能) conta

    2024年02月15日
    浏览(49)
  • 【C语言】实现动态版通讯录

    💌内容专栏:【C语言】进阶部分 💌本文概括: 结合自定义类型、动态内存管理知识,对静态版本的通讯录进行优化。 💌本文作者:花 碟 💌发布时间:2023.4.2   目录 前言: 一、静态版本代码实现: 二、动态通讯录  三、代码整理  前面我们学过了结构体、枚举等自定义

    2024年02月02日
    浏览(41)
  • C语言实践——通讯录(2)(动态版)

    首先感谢上一篇博客的大佬们的点赞,非常感谢!!! 目录 前言  一、需要添加的功能 1. 初始化——动态内存开辟 2.添加联系人——通讯录扩容 3.退出通讯录——通讯录销毁 二、具体操作 1.铺垫 2.修改初始化函数 3.修改添加函数  4.退出通讯录,新增销毁函数 上一篇文章我

    2023年04月08日
    浏览(33)
  • C语言动态内存练习:【通讯录(动态内存版本)实现】

    前面我们写了一个静态数组版本的通讯录,再结合刚学习的动态内存管理的知识,我们现在来实现一个动态内存版本的通讯录。 动态内存版本的通讯录,主要还是为了解决静态数组版本的通讯录空间太大导致的内存浪费和空间太小不够存放的问题。 扩容策略: 为通讯录设置

    2023年04月13日
    浏览(69)
  • 动态通讯录——C语言【详解+全部码源】

    作者简介: 辭七七,目前大一,正在学习C/C++,Java,Python等 作者主页: 七七的个人主页 文章收录专栏: 进阶C语言,本专栏主要讲解数据存储,进阶指针,动态内存管理,文件操作,程序环境和预处理等 欢迎大家点赞 👍 收藏 ⭐ 加关注哦!💖💖 我们之前以及写过静态版

    2023年04月21日
    浏览(38)
  • 【C语言实战项目】通讯录(动态增容版)

    🦄 个人主页 :修修修也 🎏 所属专栏 :数据结构 ⚙️ 操作环境 : Visual Studio 2022 目录 一.动态增容版简介 二.动态增容版逐步实现详解 1.创建通讯录 2. 初始化通讯录 3.增加联系人 4.通讯录增容 5.销毁通讯录 三.动态增容版代码整合 contact.c test.c contact.h 上篇博客我们一起用C语言

    2024年02月11日
    浏览(39)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包