首页 > 其他分享 >C语言实现通讯录(动态的版本)

C语言实现通讯录(动态的版本)

时间:2025-01-07 17:01:12浏览次数:3  
标签:count C语言 pc Contact 通讯录 版本 printf data

通讯录的实现框架

动态的版本
通讯录默认能存放3个人的信息
如果空间不够了,就增加空间,每次增加2个人的空间

实现一个通讯录 :
人的信息:
名字 + 年龄 + 性别 + 电话 + 地址
1.增加联系人
2.删除指定联系人
3.查找联系人
4.修改联系人
5.显示联系人
6.排序

测试功能 test.c
通讯录相关的实现 contact.c
通讯录相关的声明 contact.h

其实只是在静态的基础上稍作修改,所以这里只说明修改的部分

test.c的代码

#include "contact.h"

enum Option
{
	Exit,
	Add,
	Del,
	Search,
	Modify,
	Show,
	Sort
};

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");
}

int main()
{
	Contact con;//通讯录
	InitContact(&con);
	int input = 0;
	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:
			SortContact(&con);
			break;
		case Exit:
			DestoryContact(&con);
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误,请重新选择\n");
			break;
		}
	} while (input);
	return 0;
}

contact.h的代码

#pragma once

#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
#include<errno.h>

#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 13
#define MAX_ADDR 30
#define DEFAULT_SZ 3
#define INC_SZ 2

//通讯录函数的声明

//人的信息
typedef struct PeoInfo
{
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
}PeoInfo;


//通讯录
typedef struct Contact
{
	PeoInfo* data;
	int count;
	int capacity;
}Contact;

//初始化通讯录
void InitContact(Contact* pc);

//销毁通讯录
void DestoryContact(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 SortContact(Contact* pc);

contact.c的代码

#include "contact.h"

void InitContact(Contact* pc)
{
	assert(pc);

	pc->count = 0;
	pc->data = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));
	if (pc->data == NULL)
	{
		printf("InitContact::%s\n", strerror(errno));
		return;
	}
	pc->capacity = DEFAULT_SZ;
}

void DestoryContact(Contact* pc)
{
	assert(pc);

	free(pc->data);
	pc->data = NULL;
}

static void CheckCapacity(Contact* pc)
{
	assert(pc);

	if (pc->count == pc->capacity)
	{
		PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * (sizeof(PeoInfo)));
		if (ptr == NULL)
		{
			printf("AddContact::%s\n", strerror(errno));
			return;
		}
		pc->data = ptr;
		pc->capacity += INC_SZ;
		printf("增容成功\n");
	}
}

void AddContact(Contact* pc)
{
	assert(pc);

	//增容
	CheckCapacity(pc);

	printf("开始增加联系人\n");

	printf("请输入联系人名字:>");
	scanf("%s", pc->data[pc->count].name);
	printf("请输入联系人年龄:>");
	scanf("%d", &pc->data[pc->count].age);
	printf("请输入联系人性别:>");
	scanf("%s", pc->data[pc->count].sex);
	printf("请输入联系人电话:>");
	scanf("%s", pc->data[pc->count].tele);
	printf("请输入联系人地址:>");
	scanf("%s", pc->data[pc->count].addr);

	printf("增加成功\n");

	pc->count++;
}

void ShowContact(const Contact* pc)
{
	assert(pc);

	printf("%-20s\t%-5s\t%-5s\t%-13s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");

	int i = 0;
	for (i = 0; i < pc->count; i++)
	{
		printf("%-20s\t%-5d\t%-5s\t%-13s\t%-30s\n", pc->data[i].name,
													pc->data[i].age,
													pc->data[i].sex,
													pc->data[i].tele,
													pc->data[i].addr);
	}
}

static int FindByName(const Contact* pc, char name[])
{
	assert(pc);

	int i = 0;
	for (i = 0; i < pc->count; i++)
	{
		if (0 == strcmp(pc->data[i].name, name))
		{
			return i;
		}
	}
	return -1;
}

void DelContact(Contact* pc)
{
	assert(pc);

	if (pc->count == 0)
	{
		printf("通讯录中没有联系人,无法删除\n");
		return;
	}

	char name[MAX_NAME];
	printf("请输入要删除人的名字:>");
	scanf("%s", name);

	//删除
	//1.查找
	int pos = FindByName(pc,name);
	if (pos == -1)
	{
		printf("通讯录中没有该联系人\n");
		return;
	}
	//2.删除
	int i = 0;
	for (i = pos; i < pc->count - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}

	pc->count--;
	printf("删除成功\n");
}

void SearchContact(const Contact* pc)
{
	assert(pc);

	if (pc->count == 0)
	{
		printf("通讯录中没有联系人,无法查找\n");
		return;
	}

	char name[MAX_NAME];
	printf("请输入要查找人的名字:>");
	scanf("%s", name);

	//查找
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("通讯录中没有该联系人\n");
		return;
	}

	printf("%-20s\t%-5s\t%-5s\t%-13s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");

	printf("%-20s\t%-5d\t%-5s\t%-13s\t%-30s\n", pc->data[pos].name,
			pc->data[pos].age,
			pc->data[pos].sex,
			pc->data[pos].tele,
			pc->data[pos].addr);

}

void ModifyContact(Contact* pc)
{
	assert(pc);

	if (pc->count == 0)
	{
		printf("通讯录中没有联系人,无法修改\n");
		return;
	}

	char name[MAX_NAME];
	printf("请输入要修改人的名字:>");
	scanf("%s", name);

	//查找
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("通讯录中没有该联系人\n");
		return;
	}

	printf("%-20s\t%-5s\t%-5s\t%-13s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");

	printf("%-20s\t%-5d\t%-5s\t%-13s\t%-30s\n", pc->data[pos].name,
		pc->data[pos].age,
		pc->data[pos].sex,
		pc->data[pos].tele,
		pc->data[pos].addr);

	//修改

	enum People
	{
		Exit,
		Name,
		Age,
		Sex,
		Tele,
		Addr
	};

	int input = 0;
	do
	{
		printf("****************************\n");
		printf("**** 1.name       2.age ****\n");
		printf("**** 3.sex        4.tele****\n");
		printf("**** 5.addr       0.exit****\n");
		printf("****************************\n");
		printf("请选择要修改的信息:>");
		scanf("%d", &input);
		switch (input)
		{
		case Name:
			printf("请输入要改成的名字:>");
			scanf("%s", pc->data[pos].name);
			break;
		case Age:
			printf("请输入要改成的年龄:>");
			scanf("%d", &pc->data[pos].age);
			break;
		case Sex:
			printf("请输入要改成的性别:>");
			scanf("%s", pc->data[pos].sex);
			break;
		case Tele:
			printf("请输入要改成的电话:>");
			scanf("%s", pc->data[pos].tele);
			break;
		case Addr:
			printf("请输入要改成的地址:>");
			scanf("%s", pc->data[pos].addr);
			break;
		case Exit:
			printf("退出修改\n");
			break;
		default:
			printf("选择错误,请重新选择\n");
			break;
		}
	} while (input);

	printf("修改成功\n");
}

static int cmp_peo_by_name(const void* e1, const void* e2)
{
	return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}

void SortContact(Contact* pc)
{
	assert(pc);

	qsort(pc->data, pc->count, sizeof(PeoInfo), cmp_peo_by_name);

	printf("排序成功\n");
}

修改的部分

从无到有的修改是有思路有目的的修改,而不是先修改test.c,再修改contact.h,
最后修改contact.c,下面是从头到尾的修改思路。

contact.h中通讯录的修改

typedef struct Contact
{
	PeoInfo* data;
	int count;
	int capacity;
}Contact;

将data数组改成指针,在该指针指向的空间中进行扩容,以达到动态的目的
capacity是容量,记录到达容量限制时对data指向的空间进行扩容

contact.c中InitContact的修改

void InitContact(Contact* pc)
{
	assert(pc);

	pc->count = 0;
	pc->data = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));
	if (pc->data == NULL)
	{
		printf("InitContact::%s\n", strerror(errno));
		return;
	}
	pc->capacity = DEFAULT_SZ;
}

calloc库函数可以在堆区申请空间,并将该空间的内容初始化为0
https://legacy.cplusplus.com/reference/cstdlib/calloc/?kw=calloc
申请失败返回空指针,这时我们打印错误信息
strerror和errno也可以在cplusplus上了解
起始容量是3,所以申请了3个PeoInfo空间,并将容量初始化为3

#define DEFAULT_SZ 3

在contact.h中定义了

contact.c中AddContact的修改

static void CheckCapacity(Contact* pc)
{
	assert(pc);

	if (pc->count == pc->capacity)
	{
		PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * (sizeof(PeoInfo)));
		if (ptr == NULL)
		{
			printf("AddContact::%s\n", strerror(errno));
			return;
		}
		pc->data = ptr;
		pc->capacity += INC_SZ;
		printf("增容成功\n");
	}
}

void AddContact(Contact* pc)
{
	assert(pc);

	//增容
	CheckCapacity(pc);

	printf("开始增加联系人\n");

	printf("请输入联系人名字:>");
	scanf("%s", pc->data[pc->count].name);
	printf("请输入联系人年龄:>");
	scanf("%d", &pc->data[pc->count].age);
	printf("请输入联系人性别:>");
	scanf("%s", pc->data[pc->count].sex);
	printf("请输入联系人电话:>");
	scanf("%s", pc->data[pc->count].tele);
	printf("请输入联系人地址:>");
	scanf("%s", pc->data[pc->count].addr);

	printf("增加成功\n");

	pc->count++;
}

这里只是增加了对是否需要增容的判断
realloc库函数是增容的,cplusplus具体了解

#define INC_SZ 2

对增容的个数的定义

销毁通讯录DestoryContaxt

case Exit:
			DestoryContact(&con);
			printf("退出通讯录\n");

退出通讯录的时候,释放申请的空间

//销毁通讯录
void DestoryContact(Contact* pc);

contact.h中的声明

void DestoryContact(Contact* pc)
{
	assert(pc);

	free(pc->data);
	pc->data = NULL;
}

contact.c中的实现
在这里插入图片描述
请添加图片描述
感谢支持!

标签:count,C语言,pc,Contact,通讯录,版本,printf,data
From: https://blog.csdn.net/2402_89035880/article/details/144931074

相关文章

  • 【Prometheus】【Alertmanager】手把手教你安装v0.28.0-rc.0版本
    本文用来演示下载安装Alertmanager的v0.28.0-rc.0版本(Linux64位),具体可以按照以下步骤进行:1.下载Alertmanager压缩包首先,你需要通过curl或wget命令来下载alertmanager的压缩包。使用curl命令下载:curl-LOhttps://github.com/prometheus/alertmanager/r......
  • C语言实现对函数求积分(终端输入函数型)
    本文只提供源码,不会进行代码讲解,如需部分讲解,请前往我的另一篇文章C语言对各类复杂函数精确求导(在终端解析字符串并求导)_c语言导数函数-CSDN博客支持+-*/exp()sin()cos()tan()cot()^arc函数 log(底数,真数)输入格式为:JF<函数>dx#include<stdio.h>#include<......
  • 新一代python版本与依赖管理工具 uv
    uvpython版本与依赖管理选择原因1.新的好玩、rust开发的执行速度快2.确实方便3.轻,软件就十几二十兆,占磁盘小4.uv.lock可以锁版本场景应用场景一从anaconda、conda、pipenv等迁移过来导出依赖列表pipfreeze>requirements.txt#调整绝对路径的模块进入......
  • C语言指针数组
    1. 指针数组的定义指针数组是数组,其元素为指针。定义格式为 数据类型*数组名[数组大小] 。例如,定义一个能存放5个指向 int 类型数据的指针数组: cint*ptrArray[5];  2. 初始化-方式一:直接初始化 cintnum1=10,num2=20;int*ptrArray[]={&num......
  • Eureka加密 及Gateway搭建 - 基于SpringBoot不同版本配置方式
    前言:在最近开发过程中偶遇根据SpringBoot不同版本导致配置方式不一致等问题通用:Eureka服务配置Server端:一、引入依赖spring-boot-starter-security<dependencies> <!--Eureka声明依赖--><dependency><groupId>org.springframework.cloud</groupId>......
  • Idea兼容多版本jdk
    最近开发新项目,jdk版本也打算升级到17了,但问题在于对于老项目的迭代使用的依然是jdk8,如何解决不同版本jdk的兼容问题。将环境配置,更改为17,报如下错误:Classcom.sun.tools.javac.tree.JCTree$JCImportdoesnothavememberfield'com.sun.tools.javac.tree.JCTreequalid'......
  • 如何同时管理多个版本的node
    在前端开发中,同时管理多个版本的Node.js是一个常见的需求,特别是当你不同的项目依赖于不同版本的Node.js时。以下是一些建议和实践,帮助你有效地管理多个版本的Node.js:1.使用NodeVersionManager(nvm)nvm是一个强大的Node.js版本管理工具,它允许你在同一台机器上安装和管理多个N......
  • 如何在Windows IIS 7.5或以上版本中配置ThinkPHP的伪静态规则?
    请将以下代码另存为web.config文件,注意后缀是.config,可以先保存在记事本中,重命名,然后上传到网站根目录中,即可生效。<?xmlversion="1.0"encoding="UTF-8"?><configuration><system.webServer><rewrite><rules>&l......
  • 如何在 Windows IIS 7.5 或以上版本中配置 ThinkPHP 的伪静态规则?
    URL优化是提升用户体验和搜索引擎友好性的重要手段。ThinkPHP框架以其简洁高效的路由机制深受开发者喜爱,但在WindowsIIS7.5或更高版本的服务器环境中,正确配置伪静态规则对于实现URL优化至关重要。以下是详细的配置步骤:配置步骤创建 web.config 文件打开文本编辑器......
  • C语言的学习路线
    C语言的学习路线C语言是一种广泛使用的编程语言,它以其高效、灵活和接近底层的特性被广泛应用于系统编程、嵌入式系统、操作系统和众多应用程序开发中。对于初学者来说,学习C语言不仅是了解计算机编程的一个良好起点,同时也为后续学习其他编程语言打下坚实的基础。本文将详细......