当前位置:网站首页>C语言版通讯录——动态存储(进阶版)

C语言版通讯录——动态存储(进阶版)

2022-08-11 05:16:00 云逸943

前言

        之前,我出了一期关于通讯录管理系统的项目实现,里面可以实现通讯录人员的增添改查显示以及排序统计功能,但那种只是以静态开辟内存的方式进行编写,大小固定死,若是达到上限便不可以再进行添加。这次优化,我会将静态存储改为动态存储,达到可以永久性添加通讯录人员的功能实现。

        注:这里需要用到之前的静态版本的代码: C语言版通讯录操作系统 

                我就不从头开始编写对代码的实现,只是展示动态实现的代码。

        

        动态存储要求:通讯录开始的人员为空,容量为3个,若通讯录达到容量上限,且等到再一次新增人员时,程序会自动进行扩容,每次扩容+2。

一.部分代码改进:

1结构体:


typedef struct PeoInf {
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tale[MAX_TALE];
	char addr[MAX_ADDR];
}PeoInf;


通讯录结构体(静态版本)
typedef struct Contact {
	struct PeoInf data[MAX];//可以存放100个通讯录成员
	int count;
}Contact;


                                                                改进为:

typedef struct PeoInf {
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tale[MAX_TALE];
	char addr[MAX_ADDR];
}PeoInf;

//通讯录结构体(动态版本)
typedef struct Contact {
	struct PeoInf* data;
	int count;
	int capacity;//容量
}Contact;

        将静态版的通讯录结构体数组成员改为指针成员,另外新增了一个成员capacity,意为容量,当添加的成员达到容量上限,程序便可自动进行扩充容量,提升效率! 


2.初始化通讯录函数:

//静态版初始化通讯录
void InitContact(Contact* pc) {
	pc->count = 0;
	memset(pc->data, 0, sizeof(PeoInf));//初始化数组全为0
}

                                

                                                                改进为: 

//动态版通讯录初始化
int InitContact(Contact* pc) {
	assert(pc);
	pc->count = 0;
	pc->capacity = INIT_CA;//初始能放3(容量)
	pc->data = (PeoInf*)calloc(INIT_CA , sizeof(PeoInf));
	if (pc->data == NULL) {
		printf("%s\n", strerror(errno));
		return -1;
	}
	return 0;
}

        通过对指针接收calloc函数动态开辟的内存。

        注:1.初始化的通讯录数据全为0,capacity成员变量开始可放下3个成员的信息属性 。

                2.开辟之后的内存空间由指针接收,且进行是否成功开辟内存空间的判断,若开辟失败,则会报错,终止程序进行。


3.增添成员功能:

函数功能实现(静态版本)
void AddContact(Contact* pc) {
	assert(pc);
	if (pc->count == 100) {
		printf("通讯录已满,无法再添加人员\n");
		return;
	}
	else {
		printf("正在准备添加验证,请稍后:\n");
		printf("请输入要添加人的姓名:\n");
		scanf("%s", pc->data[pc->count].name);
		printf("请输入要添加人的年龄:\n");
		scanf("%d", &(pc->data[pc->count].age));
		printf("请输入要添加人的性别:\n");
		scanf("%s", pc->data[pc->count].sex);
		printf("请输入要添加人的电话:\n");
		scanf("%s", pc->data[pc->count].tale);
		printf("请输入要添加人的地址:\n");
		scanf("%s", pc->data[pc->count].addr);
		pc->count++;
		printf("添加成功\n");
	}
}

                                                                改进为:

void check_Capacity(Contact* pc) {//扩容函数
	if (pc->count == pc->capacity) {
		PeoInf* ptr = realloc(pc->data, (pc->capacity + ADD_CA) * sizeof(PeoInf));//动态分配
		if (ptr == NULL) {
			printf("%s\n", strerror(errno));
			return;
		}
		else {	
			pc->data = ptr;
			pc->capacity += ADD_CA;
			printf("增容成功\n");
		}
	}
}

//增添功能(动态版)
void AddContact(Contact* pc) {
		assert(pc);
		check_Capacity(pc);//扩容函数
	
			printf("正在准备添加验证,请稍后:\n");
			printf("请输入要添加人的姓名:\n");
			scanf("%s", pc->data[pc->count].name);
			printf("请输入要添加人的年龄:\n");
			scanf("%d", &(pc->data[pc->count].age));
			printf("请输入要添加人的性别:\n");
			scanf("%s", pc->data[pc->count].sex);
			printf("请输入要添加人的电话:\n");
			scanf("%s", pc->data[pc->count].tale);
			printf("请输入要添加人的地址:\n");
			scanf("%s", pc->data[pc->count].addr);
			pc->count++;
			printf("添加成功\n");
		}

        增添成员的函数功能中,新增了一个扩容函数,需要使用到realloc函数。每当增加的成员达到容量上限时,就会自动进行扩容,每次扩容+2,可以放下两个人员的信息属性。 


4.释放动态开辟的内存

        在使用动态开辟的内存空间时,它是在堆区被使用的,所以必须使用free函数去释放之前开辟的内存空间,否则会造成内存泄漏,降低程序使用效率。(情况很严重!!!) 

        

void FreeContact(Contact* pc) {
	assert(pc);
	free(pc->data);
	pc->data = NULL;
}

case EXIT:
			//释放动态内存
			FreeContact(&con);
			printf("即将退出程序\n");
			break;

二.完整代码:

1.test.c代码实现:

#define _CRT_SECURE_NO_WARNINGS 1
# include<stdio.h>
#include"Contact.h"

enum GN {
	EXIT,
	ADD,
	DEL,
	MOD,
	SEARCH,
	SHOW,
	QSORT,
    Total
};

void menu() {
	printf("****************欢迎来到通讯录管理界面********************\n");
	printf("*************************1.Add.*************************\n");
	printf("*************************2.Del.*************************\n");
	printf("*************************3.Mod.*************************\n");
	printf("************************4.Search.***********************\n");
	printf("************************5.Show.*************************\n");
	printf("************************6.Sort.*************************\n");
    printf("************************7.Total.************************\n");
	printf("************************0.Exit.*************************\n");

	}
int main() {

	 Contact con;//创建通讯录对象

	//初始化通讯录
	InitContact(&con);

	int input = 0;
	do {
		menu();
		printf("请输入你的选择:\n");
		scanf("%d", &input);
		switch (input) {
		case ADD:
			AddContact(&con);
			break;
		case DEL:
			DelContact(&con);
			break;
		case MOD:
			ModfiyContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case QSORT:
			QsortContact(&con);
			break;
		case EXIT:
			//释放动态内存
			FreeContact(&con);
			printf("即将退出程序\n");
			break;
        case Total:
            TotContact(&con);
            break;
		default:
			printf("输入有误!\n");
			break;
		}
	} while (input);
	return 0;
}

2.Contact.c函数功能代码实现:

#include"Contact.h"



//动态版通讯录初始化
int InitContact(Contact* pc) {
	assert(pc);
	pc->count = 0;
	pc->capacity = INIT_CA;//初始能放3(容量)
	pc->data = (PeoInf*)calloc(INIT_CA , sizeof(PeoInf));
	if (pc->data == NULL) {
		printf("%s\n", strerror(errno));
		return -1;
	}
	return 0;
}




void check_Capacity(Contact* pc) {//扩容函数
	if (pc->count == pc->capacity) {
		PeoInf* ptr = realloc(pc->data, (pc->capacity + ADD_CA) * sizeof(PeoInf));//动态分配
		if (ptr == NULL) {
			printf("%s\n", strerror(errno));
			return;
		}
		else {	
			pc->data = ptr;
			pc->capacity += ADD_CA;
			printf("增容成功\n");
		}
	}
}

//增添功能(动态版)
void AddContact(Contact* pc) {
		assert(pc);
		check_Capacity(pc);//扩容函数
	
			printf("正在准备添加验证,请稍后:\n");
			printf("请输入要添加人的姓名:\n");
			scanf("%s", pc->data[pc->count].name);
			printf("请输入要添加人的年龄:\n");
			scanf("%d", &(pc->data[pc->count].age));
			printf("请输入要添加人的性别:\n");
			scanf("%s", pc->data[pc->count].sex);
			printf("请输入要添加人的电话:\n");
			scanf("%s", pc->data[pc->count].tale);
			printf("请输入要添加人的地址:\n");
			scanf("%s", pc->data[pc->count].addr);
			pc->count++;
			printf("添加成功\n");
		}


//展示
void ShowContact(Contact* pc) {
	assert(pc);
	if (pc->count == 0) {
		printf("通讯录为空,无法查看成员信息\n");
		return;
	}
	else {
		int i = 0;
		printf("%-5s\t%-4s\t%-5s\t%-12s\t%-20s\n", "姓名", "年龄", "性别", "电话", "住址");
		for (i = 0; i < pc->count; i++) {
			printf("%-5s\t%-4d\t%-5s\t%-12s\t%-20s\n", pc->data[i].name,
				pc->data[i].age,
				pc->data[i].sex,
				pc->data[i].tale,
				pc->data[i].addr);
		}
	}
}

//查找
int Find(Contact* pc, char name[]) {
	assert(pc);
	int i = 0;
	for (i = 0; i < pc->count; i++) {
		if (strcmp(pc->data[i].name, name) == 0) {
			return i;
		}
	}
	return -1;
}


//删除
void DelContact(Contact* pc) {
	char name[MAX_NAME] = { 0 };
	assert(pc);
	if (pc->count == 0) {
		printf("通讯录为空,无法进行删除操作\n");
		return;
	}
	else {
		printf("请输入要删除人的姓名:\n");
		scanf("%s", name);
		int ret = Find(pc, name);
		if (ret == -1) {
			printf("要查找的人不存在,无法删除\n");
			return;
		}
		else {
			printf("正在准备删除操作验证:\n");
			int i = 0;
			for (i = ret; i < pc->count - 1; i++) {
				pc->data[i] = pc->data[i + 1];//删除后所有成员都要向前移动
			}
			pc->count--;
			printf("删除成功\n");
		}
	}
}
 
//修改功能
void ModfiyContact(Contact* pc) {
	assert(pc);
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改的人的姓名:\n");
	scanf("%s", name);

	if (pc->count == 0) {
		printf("通讯录为空,无法修改\n");
		return;
	}
	else {
		int ret = Find(pc, name);
		if (ret == -1) {
			printf("要修改的人员名字不存在\n");
			return;
		}
		else {
			printf("正在准备修改验证操作:\n");
			printf("请输入要修改人的姓名:\n");
			scanf("%s", pc->data[ret].name);
			printf("请输入要修改人的年龄:\n");
			scanf("%d", &(pc->data[ret].age));
			printf("请输入要修改人的性别:\n");
			scanf("%s", pc->data[ret].sex);
			printf("请输入要修改人的电话:\n");
			scanf("%s", pc->data[ret].tale);
			printf("请输入要修改人的地址:\n");
			scanf("%s", pc->data[ret].addr);
			printf("修改成功\n");
		}
	}
}

//查找功能
void SearchContact(Contact* pc) {
	char name[MAX_NAME] = { 0 };
	assert(pc);
	if (pc->count == 0) {
		printf("查找失败,通讯录为空,请先添加\n");
		return;
	}
	else {
		printf("请输入要查找的人的姓名:\n");
		scanf("%s", &name);
		int ret = Find(pc, name);
		if (ret == -1) {
			printf("要查找的人不存在\n");
			return;
		}
		else {
			printf("%-5s\t%-4s\t%-5s\t%-12s\t%-20s\n", "姓名", "年龄", "性别", "电话", "住址");
				printf("%-5s\t%-4d\t%-5s\t%-12s\t%-20s\n", pc->data[ret].name,
					pc->data[ret].age,
					pc->data[ret].sex,
					pc->data[ret].tale,
					pc->data[ret].addr);
				printf("查找成功\n");
		}
	}
}


int cmp_by_name(const void* e1, const void* e2) {
	return strcmp(((PeoInf*)e1)->name, ((PeoInf*)e2)->name);
	}

int cmp_by_age(const void* e1, const void* e2) {
	return ((PeoInf*)e1)->age-((PeoInf*)e2)->age;
}

//排序
void QsortContact(Contact* pc) {
	assert(pc);
	if (pc->count == 0) {
		printf("通讯录为空,无法排序\n");
		return;
	}
	else {
		int i = 0;
		printf("1.按姓名排序      2.按年龄排序    \n");
		printf("请输入你想排序的序号:\n");
		scanf("%d", &i);
		if(i==1)
		qsort(pc->data, pc->count, sizeof(PeoInf), cmp_by_name);

		if (i == 2)
		qsort(pc->data, pc->count, sizeof(PeoInf), cmp_by_age);

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

void FreeContact(Contact* pc) {
	assert(pc);
	free(pc->data);
	pc->data = NULL;
}

	void TotContact(Contact* pc) {
		assert(pc);
		if (pc->count == 0) {
			printf("通讯录为空,人员为空\n");
			return;
		}
		else {
			int total = pc->count;
			printf("目前通讯录中共有%d名联系人\n", total);
		}
	}

3.Contact.h头文件代码实现:

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
# 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_TALE 12
#define MAX_ADDR 20
#define MAX_ 100

#define INIT_CA 3 //(初始化时,通讯录容量为3)
#define ADD_CA 2  //当容量满了后,每增加一个成员就要多开辟两个内存量

//通讯录成员信息属性
typedef struct PeoInf {
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tale[MAX_TALE];
	char addr[MAX_ADDR];
}PeoInf;


//通讯录结构体(静态版本)
//typedef struct Contact {
//	struct PeoInf data[MAX];//可以存放100个通讯录成员
//	int count;
//	int capacity;
//}Contact;


//通讯录结构体(动态版本)
typedef struct Contact {
	struct PeoInf* data;
	int count;
	int capacity;//容量
}Contact;






int InitContact(Contact* pc);

void AddContact(Contact* pc);

void ShowContact(Contact* pc);

void DelContact(Contact* pc);

void ModfiyContact(Contact* pc);

void SearchContact(Contact* pc);

void QsortContact(Contact* pc);

void TotalContact(Contact* pc);

//释放动态内存
void FreeContact(Contact* pc);

 

 以上就是动态存储通讯录操作系统的完整实现了!

原网站

版权声明
本文为[云逸943]所创,转载请带上原文链接,感谢
https://blog.csdn.net/weixin_69283129/article/details/126133901