c语言之动态版通讯录

本文介绍如何使用C语言实现一个动态通讯录系统,能够灵活添加、删除、查找、修改联系人信息,支持按姓名排序及清空联系人列表。重点讨论了内存管理函数malloc和realloc的使用,以及结构体与枚举类型在实现通讯录功能中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

实现一个通讯录,可以存储1000个人的信息。每个人的信息包括姓名,电话,年龄,性别,住址。

要求如下:

1. 添加联系人信息
2. 删除指定联系人信息
3. 查找指定联系人信息
4. 修改指定联系人信息
5. 显示所有联系人信息
6. 清空所有联系人
7. 以名字排序所有联系人

因为是动态版本,所以最大联系人的数目不在是确定的。需要动态开辟。这里要用malloc或realloc开辟动态空间。这里先讲一下malloc 和 realloc的基本用法

1.malloc

函数原型:extern void *malloc(unsigned int num_bytes);

头文件:#include <stdlib.h或者#include <malloc.h>

函数功能:在内存中动态分配一块num_bytes大小的内存空间。malloc()函数会返回一个指针,该指针的指向分配的内存空间,如果出现错误返回NULL。 

注意:1.malloc()函数分配的内存空间是在堆上进行的。所以用完后程序员要用free()函数释放掉。

          2.void* 表示未确定类型的指针,void*可以指向任何类型的数据。也就是说这块申请的内存空间还不知道要存储什么类型的数据,可以根据自己的需要强制转换为自己需要的类型。

举例1:

int main(void)
{
	int *parr;
	parr = (int*)malloc(20*sizeof(parr));//开辟80字节的大小
	int i = 0;
	for (i = 0; i < 20; i++)
	{
		parr[i] = i;
	}
	for (i = 0; i < 20; i++)
	{
		printf("%d ", parr[i]);
	}
	free(parr);
	parr = NULL;//防止生成野指针
	system("pause");
	return 0;
}

2.realloc

函数原型: extern void* realloc(void*men_address,unsigned int newsize)

语法:  指针名=(数据类型)realloc (要改变内存大小的指针名,扩展后的大小(不是新添加的大小));

注:需要开辟的大小可大可小,若果新的大小大于原内存大小,那么新分配的空间不会被初始化。如果新的大小小于原内存大小,可能会导致数据丢失。

头文件:#include<stdlib.h>

功能:先判断当前指针是否有足够的连续空间,如果有,扩大men_address之前的地址,并且将men_address返回 。如果空间不够,先按照newsize指定的大小分配空间,将原有的数据从头到尾拷贝到新分配的内存中去,然后释放原来的men_address所指向的内存区。注意原来的指针会自动释放,不需要使用   free(),同时返回新分配的内存区域的首地址(即重新分配内存块)。

返回值:成功分配内存返回指向被分配的首地址的指针。失败返回空指针NULL。

int main()
{
	int *p;
	char *ch;
	ch = (char*)malloc(sizeof(char));//用malloc 开辟一个字节的空间
	printf("%d\n", sizeof(char));
	p = (int*)realloc(ch, sizeof(int));//用realloc开辟新的四个字节的空间
	printf("%d\n", sizeof(p));
	system("pause");
	return 0;
}

 

正文:

contact.h 

这个头文件包含了一个关于人信息的结构体,枚举了各种需要实现的功能,和各个函数的申明。

#ifndef __CONTACT_H_
#define __CONTACT_H_

#define _CRT_SECURE_NO_WARNINGS
#define MAX_NAME 20
#define MAX_SEX 10
#define MAX_ADDR 30
#define MAX_TEL 15
#define MAX 1000//定义最大联系人的数量
#define DEFAULT_SZ 3


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

enum option//通讯录的各种功能,严格按照菜单的顺序写,否则会出错
{
	EXIT,
	ADD,
	DEL,
	SEARCH,
	MODIFY,
	SHOW,
	SORT,
	EMPTY,
};
typedef struct Peoinfo//定义人的信息
{
	char name[MAX_NAME];
	short age;
	char sex[MAX_SEX];
	char addr[MAX_ADDR];
	int  tel[MAX_TEL];
}Peoinfo,*pPeoinfo;//pPeoinfo是指向Peoinfo结构体的一个指针

typedef struct contact
{
	pPeoinfo data;//存放数据的位置
	int sz;//有效元素的个数
	int capacity;//容量
}contact, *pcontact;

void InitContact(pcontact pc); //初始化信息
void DistroyContact(pcontact pc);//销毁通讯录
void AddContact(pcontact pc);//添加联系人
void ShowContact(pcontact pc);//显示联系人
void DelContact(pcontact pc);//删除联系人
void ModifyContact(pcontact pc);//修改联系人
void EmptyContact(pcontact pc);//置空联系人
void SortContact(pcontact pc);//排序联系人
int SearchContact(pcontact pc);//查找联系人
void CheckCapacity(pcontact pc);//判断是否增容
#endif

 

因为是动态版本,所以最大联系人的数目是不确定的,需要用realloc来扩容。在最开始也会用malloc开辟默认大小的容量去存储联系人。所以在添加联系人时,需要判断是否扩容。当容量的大小和联系人的数目相同时,才需要扩容,所以这里需要一个检查容量,扩容函数的CheckCapacityt()函数。在重要的地方做了注释可以参考参考

 

#include"contact.h"
void InitContact(pcontact pc)//初始化
{
	pc->sz = 0;
	pc->capacity = DEFAULT_SZ;//最开始先给定一个容量去存储联系人的信息
	pc->data = (pPeoinfo)malloc(pc->capacity * sizeof(Peoinfo));//动态开辟默认容量大小的字节
	if (pc->capacity == pc->sz)//判断是否需要增容
	{
		printf("%s\n", strerror(errno));
		exit(EXIT_FAILURE);//若果增容失败就退出程序
	}
}
void DistroyContact(pcontact pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->data = 0;
	pc->sz = 0;
}
void CheckCapacity(pcontact pc)//检查容量,若容量不够要扩容
{
	if (pc->capacity == pc->sz)//若容量等于联系人个数,就需要扩容
	{
		pPeoinfo ptr = realloc(pc->data, (pc->capacity += 2)*sizeof(Peoinfo));
		if (ptr != NULL)//判断是否增容成功
		{
			pc->data = ptr;
			printf("增容成功");
		}
	}
}
void AddContact(pcontact pc)//添加联系人
{
	//先检查容量是否足够,若不够要扩容
	CheckCapacity(pc);
	printf("请输入姓名>");
	scanf("%s", pc->data[pc->sz].name);
	//pc->data为数组名,pc->sz为数组的下标。通过.访问数组成员
	printf("请输入电话>");
	scanf("%s", pc->data[pc->sz].tel);
	printf("请输入年龄>");
	scanf("%d", &pc->data[pc->sz].age);
	printf("请输入性别>");
	scanf("%s", pc->data[pc->sz].sex);
	printf("请输入地址>");
	scanf("%s", pc->data[pc->sz].addr);
	(pc->sz)++;
	printf("添加成功\n");
}
void ShowContact(pcontact pc)//显示所有的联系人
{
	int i = 0;
	printf("%-10s\t%-10s%-5s\t%-5s\t%-10s\n", "name", "tel", "age", "sex", "addr");//打印标头
	for (i = 0; i < pc->sz; i++)//打印所以来呢西人的信息
	{
		printf("%-10s\t%-10s%-5d\t%-5s\t%-10s\n",
			pc->data[i].name,
			pc->data[i].tel,
			pc->data[i].age,
			pc->data[i].sex,
			pc->data[i].addr
			);
	}
}
static int FindEntry(pcontact pc, char name[])//查找是否存在该联系人,在删除,修改联系人中会用到,存在返回i,不存在返回-1
{
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(name, pc->data[i].name) == 0)//比较联系人是否存在
		{
			return i;
		}
	}
	return-1;
}
void DelContact(pcontact pc)//删除联系人
{
	char name[MAX_NAME] = { 0 };
	int pos = 0;
	if (pc->sz == 0)//判断是否有联系人可以删除
	{
		printf("没有联系人可以删除\n");
		return;
	}
	printf("请输入要删除的联系人");
	scanf("%s", name);
	pos = FindEntry(pc, name);//查找是否存在该联系人
	if (pos == -1)
	{
		printf("没有找到该联系人\n");
		return;
	}
	else
	{
		int i = 0;
		for (i = pos; i < pc->sz - 1; i++)
		{
			pc->data[i] = pc->data[i + 1];//让后面的联系人覆盖该联系人,达到删除的作用
		}
		(pc->sz)--;//联系人的下标-1
		printf("删除成功\n");
	}
}
void EmptyContact(pcontact pc)//置空联系人
{

	pc->sz = 0;//清空联系人只需要让pc->sz=0就好,这样里面的数据都没了 。
	printf("清空联系人成功\n");
}
int SearchContact(pcontact pc)//查找联系人
{
	int pos = 0;
	char name[20];
	printf("请输入要查找的联系人\n");
	scanf("%s", &name);
	pos = FindEntry(pc, name);//查找是否存在该联系人
	if (pos == -1)
	{
		printf("不存在该联系人\n");
	}
	else
	{
		printf("姓名:%s\n", pc->data[pos].name);
		printf("电话:%s\n", pc->data[pos].tel);
		printf("年龄:%d\n", pc->data[pos].age);
		printf("性别:%s\n", pc->data[pos].sex);
		printf("地址:%s\n", pc->data[pos].addr);
	}
	return pos;
}
void menu1()
{
	printf("***  1.修改姓名   ***\n");
	printf("***  2.修改电话   ***\n");
	printf("***  3.修改年龄   ***\n");
	printf("***  4.修改性别   ***\n");
	printf("***  5.修改地址   ***\n");
	printf("***  0.退出修改   ***\n");
}
void ModifyContact(pcontact pc)//修改联系人
{
	char name[20] = { 0 };
	int pos = 0;
	printf("请输入要修改的联系人>");
	scanf("%s", &name);
	pos = FindEntry(pc, name);//查找是否存在该联系人

	if (pos != -1)//如果存在
	{
		printf("姓名:%s\n", pc->data[pos].name);
		printf("电话:%s\n", pc->data[pos].tel);
		printf("年龄:%d\n", pc->data[pos].age);
		printf("性别:%s\n", pc->data[pos].sex);
		printf("地址:%s\n", pc->data[pos].addr);
		int i = 0;
		do
		{

			menu1();
			printf("请输入修改项>");
			scanf("%d", &i);

			switch (i)
			{
			case 1:
				printf("把当前姓名修改为:\n");
				scanf("%s", pc->data[pos].name);
				break;
			case 2:
				printf("把当前的电话修改为:\n");
				scanf("%s", pc->data[pos].tel);
				break;
			case 3:
				printf("把当前的年龄修改为:\n");
				scanf("%d", &pc->data[pos].age);
				break;
			case 4:
				printf("把当前性别修改为:\n");
				scanf("%s", pc->data[pos].sex);
				break;
			case 5:
				printf("把当前的地址修改为:\n");
				scanf("%s", pc->data[pos].addr);
				break;
			case 0:
				break;
			default:
				printf("输入选项错误\n");
				break;
			}
		} while (i);
	}
	else
	{
		printf("不存在该联系人\n");
	}
}
void SortContact(pcontact pc)//对联系人按年龄排序
{
	int i = 0;
	int j = 0;
	for (i = 0; i < pc->sz - 1; i++)
	{
		for (j = 0; j < pc->sz - 1 - i; j++)
		{
			if (pc->data[j].age< pc->data[j + 1].age)
			{
				Peoinfo temp;//创建一个与pc->data的类型相同存放pc->data
				temp = pc->data[j];//交换两个人的信息
				pc->data[j] = pc->data[j + 1];
				pc->data[j + 1] = temp;
			}
		}
	}
	printf("联系人排序成功\n");
}

 

这一部分和静态版本的是一样的,没有做任何改变。

 

#include"contact.h"
void menu()
{
	printf("*** 1.ADD      2.DEL    ***\n");
	printf("*** 3.SEARCH   4.MODIFY ***\n");
	printf("*** 5.SHOW     6.SORT   ***\n");
	printf("*** 7.EMPTY    0.EXIT   ***\n");
}

int main()
{
	int input;
	contact my_con;
	InitContact(&my_con);
	//传地址的原因:
	//1.结构体传参要传地址 2.结构体内部要改变函数外边的值要传地址
	do
	{
		menu();
		printf("请选择> ");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContact(&my_con);
			break;
		case DEL:
			DelContact(&my_con);
			break;
		case SEARCH:
			SearchContact(&my_con);
			break;
		case MODIFY:
			ModifyContact(&my_con);
			break;
		case SHOW:
			ShowContact(&my_con);
			//可以取地址,也可以不取地址。因为只是查看不是改变内容。
			break;
		case SORT:
			SortContact(&my_con);
			break;
		case EMPTY:
			EmptyContact(&my_con);
			break;
		case EXIT:
			DistroyContact(&my_con);
			break;
		default:
			printf("输入有误");
		}
	} while (input);
	return 0;
}

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值