C语言课程设计|通讯录管理系统(文件操作版本)

一、需求分析

实现一个简单的通讯录管理系统我们需要具备以下几点基本功能:增加通讯录的联系人,删除通讯录的联系人,查找通讯录的联系人,修改通讯录的联系人,显示通讯录的联系人,清空通讯录,按指定内容排序通讯录,实现动态版本的通讯录,避免空间不足和空间的浪费,最后保存通讯录的信息。

一共分为test.c和contact.c两个源文件,一个contact.h的头文件
test.c:是实现基本的逻辑
contact.c:是功能函数的实现
contact.h :是函数的声明,结构体的声明和头文件的包含

二、通讯录的实现
1.实现通讯录的前期准备
(1)菜单的实现

菜单的实现很简单,只需要把内容封装成一个函数,然后打印出来。

void menu()
{
	printf("*************************************\n");
	printf("******** 1.add      2.dele   ********\n");
	printf("******** 3.search   4.modify ********\n");
	printf("******** 5.show     6.empty  ********\n");
	printf("******** 7.sort     0.exit   ********\n");
	printf("*************************************\n");
}
(2)实现菜单选项的逻辑

使用do…while();来控制整个逻辑的循环,里面包含switch语句,根据我们要实现的功能来选择对应的选项,选项使用枚举,我们能更好的知道我们要选择功能对应的选项是什么,这样整个菜单的逻辑就实现了。

enum Option
{
	EXIT,
	ADD,
	DELE,
	SEARCH,
	MODIFY,
	SHOW,
	EMPTY,
	SORT
};

int main()
{
	Contact con;
	InitContac(&con);
	int input = 0;
	do
	{
		menu();
		printf("请选择: ");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContacet(&con);
			break;
		case DELE:
			DeleContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case MODIFY:
			ModefyContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case EMPTY:
			EmptyContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case EXIT:
			SaveContact(&con);
			DestroyContact(&con);
			printf("退出\n");
			break;
		default:
			printf("没有此选项,请重新输入!\n");
			break;
		}

	} while (input);
	return 0;
}
(3)联系人的结构体的创建

我们使用一个结构体来创建通讯录人员的信息,每一个成员变量的大小都是使用
#define来定义的,方便后期结构体成员变量大小的调整,然后再创建一个结构体,里面有通讯录人员的信息,通讯录人员总数和通讯录的总容量,有静态版本和动态版版本的,这里使用的是动态版本的。

typedef struct PeopInfor
{
	char name[NAME_MAX];
	int age;
	char sex[SEX_MAX];
	char tele[TELE_MAX];
	char addr[ADDR_MAX];
}PeopInfor;


静态版本
//typedef struct Contact
//{
//	PeopInfor data[MAX];
//	int sz;
//}Contact;


//动态版本
typedef struct Contact
{
	PeopInfor* data;//通讯录人员的信息
	int sz;//记录通讯录有多少人的信息
	int capacity;//记录当前通讯录的总容量
}Contact;
(4)#define定义的符号

上面两个结构体创建的变量的大小都是#define定义的,方便后期对结构体成员变量大小的修改。

#define MAX 1000
#define NAME_MAX 10
#define SEX_MAX 4
#define TELE_MAX 12
#define ADDR_MAX 20

#define DEFAULT_SZ 3
#define INC_SZ 2
2.功能函数的实现
(1)初始化通讯录

下面通讯录的初始化有静态和动态版本,我们使用calloc函数来动态开辟内存空间,函数的功能是为num个大小为size的元素开辟一块空间,并把空间每个字节初始化为0;如果之前通讯录有保存过的数据,就会加载到通讯录中。

calloc函数原型:
void* calloc (size_t num,size_t size);
size_t num – num个元素
size_t size – 一个元素的大小(字节)

通讯录初始化,静态版本
//void InitContac(Contact* pc)
//{
//	assert(pc);
//	pc->sz = 0;
//	memset(pc->data, 0, sizeof(pc->data));
//}

//通讯录初始化,动态版本
void InitContac(Contact* pc)
{
	assert(pc);
	pc->sz = 0;
	PeopInfor* ptr = (PeopInfor*)calloc(DEFAULT_SZ, sizeof(PeopInfor));
	if (ptr == NULL)
	{
		perror("InitContac::calloc");
		return;
	}
	pc->data = ptr;
	pc->capacity = DEFAULT_SZ;

	LodeContact(pc);
}
//加载联系人
void LodeContact(Contact* pc)
{
	FILE* pf = fopen("contact.txt", "rb");
	if (NULL == pf)
	{
		perror("LodeContact");
	}
	else
	{
		PeopInfor temp = { 0 };
		int i = 0;
		while (fread(&temp, sizeof(PeopInfor), 1, pf))
		{
			Check_capacity(pc);
			pc->data[i] = temp;
			pc->sz++;
			i++;
		}
		fclose(pf);
		pf = NULL;
	}
}
(2)增加联系人

首先判断空间是否足够,不够则增容,我们封装成了Check_capacity函数,方便后面调用,然后增加信息,输入对应内容。

//添加通讯录人员,静态版本
//void AddContacet(Contact* pc)
//{
//	if (pc->sz == MAX)
//	{
//		printf("通讯录人员已满,无法添加\n");
//		return;
//	}
//	int num = 0;
//	printf("请输入要增加的人数:");
//	scanf("%d", &num);
//	int i = 0;
//	for (i = 0; i < num; i++)
//	{
//		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++;
//	}
//}

//检查空间是否足够,不够则增容
void Check_capacity(Contact* pc)
{
	if (pc->sz == pc->capacity)
	{
		PeopInfor* ptr = (PeopInfor*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeopInfor));
		if (ptr == NULL)
		{
			perror("Check_capacity::realloc");
			return;
		}
		pc->capacity += INC_SZ;
		pc->data = ptr;
		printf("增容成功\n");
	}
}

//添加通讯录人员,动态版本
void AddContacet(Contact* pc)
{
	assert(pc);
	Check_capacity(pc);
	int num = 0;
	printf("请输入要增加的人数:");
	scanf("%d", &num);
	int i = 0;
	for (i = 0; i < num; i++)
	{
		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);
		printf("添加成功!\n");
		pc->sz++;
	}
}

打印结果:
在这里插入图片描述

(3)显示所有联系人的信息

使用for循环打印结构体中sz个信息

//显示通讯录名单
void ShowContact(const Contact* pc)
{
	assert(pc);
	printf("%-5s\t%-4s\t%-5s\t%-15s%-20s\n",
		"姓名", "年龄", "性别", "电话", "地址");
	for (int i = 0; i < pc->sz; i++)
	{
		printf("%-5s\t%-4d\t%-5s\t%-15s%-20s\n",
			pc->data[i].name,
			pc->data[i].age,
			pc->data[i].sex,
			pc->data[i].tele,
			pc->data[i].addr);
	}
}

打印结果:
在这里插入图片描述

(4)查找指定姓名的联系人

通过输入的姓名查找对应的下标,如果找到则返回下标,没找到则返回-1,为了方便后期的调用,使用FindByName函数封装起来;用返回的下标打印查找人的信息。

int FindByName(const Contact* pc, char name[])
{
	assert(pc);
	for (int pos = 0; pos < pc->sz; pos++)
	{
		if (strcmp(pc->data[pos].name, name) == 0)
		{
			return pos;
		}
	}
	return -1;
}

//查找指定联系人
void SearchContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX];
	printf("请输入要查找人的姓名:");
	scanf("%s", name);
	int ret = FindByName(pc, name);
	if (-1 == ret)
	{
		printf("要查找的人不存在\n");
		return;
	}

	printf("%-5s\t%-4s\t%-5s\t%-15s%-20s\n",
		"姓名", "年龄", "性别", "电话", "地址");
		printf("%-5s\t%-4d\t%-5s\t%-15s%-20s\n",
			pc->data[ret].name,
			pc->data[ret].age,
			pc->data[ret].sex,
			pc->data[ret].tele,
			pc->data[ret].addr);
}

显示结果:
在这里插入图片描述

(5)修改指定姓名的联系人

使用枚举我们可以一眼就看见选项对应的功能是什么,通过指定姓名修改指定的内容,也可以修改全部内容

enum Option
{
	EXIT,
	NAME,
	AGE,
	SEX,
	TEL,
	ADDR,
	TOTAL
};

//修改指定联系人
void ModefyContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX];
	printf("请输入要修改人的姓名:");
	scanf("%s", name);
	int ret = FindByName(pc, name);
	if (-1 == ret)
	{
		printf("要修改的人不存在\n");
		return;
	}
	else
	{
		int num = 0;
		do
		{
			printf("请输入你想修改此联系人的信息:\n");
			printf("0.退出  1.姓名  2.年龄  3.性别  4.电话  5.地址  6.全部修改\n");
			scanf("%d", &num);
			switch (num)
			{
			case NAME:
				printf("请输入名字:");
				scanf("%s", pc->data[ret].name);
				printf("修改成功!\n");
				break;
			case AGE:
				printf("请输入年龄:");
				scanf("%d", &(pc->data[ret].age));
				printf("修改成功!\n");
				break;
			case SEX:
				printf("请输入性别:");
				scanf("%s", pc->data[ret].sex);
				printf("修改成功!\n");
				break;
			case TEL:
				printf("请输入电话:");
				scanf("%s", pc->data[ret].tele);
				printf("修改成功!\n");
				break;
			case ADDR:
				printf("请输入地址:");
				scanf("%s", pc->data[ret].addr);
				printf("修改成功!\n");
				break;
			case TOTAL:
				printf("请输入名字:");
				scanf("%s", pc->data[ret].name);
				printf("请输入年龄:");
				scanf("%d", &(pc->data[ret].age));
				printf("请输入性别:");
				scanf("%s", pc->data[ret].sex);
				printf("请输入电话:");
				scanf("%s", pc->data[ret].tele);
				printf("请输入地址:");
				scanf("%s", pc->data[ret].addr);
				printf("修改成功!\n");
				break;
			case EXIT:
				printf("退出\n");
				break;
			default:
				printf("无此选项\n");
				break;
			}
		} while (num);
	}
}

显示结果:
在这里插入图片描述

(6)删除指定姓名的联系人

找到指定人的下标并返回,以下标位置开始,把后面的内容向前覆盖,这样指定的位置就会删除

//删除指定联系人
void DeleContact(Contact* pc)
{
	char name[NAME_MAX];
	assert(pc);
	printf("请输入要删除人的姓名:");
	scanf("%s", name);
	if (pc->sz == 0)
	{
		printf("通讯录为空,无法删除\n");
	}
	int ret = FindByName(pc, name);
	if (-1 == ret)
	{
		printf("要删除的人不存在\n");
		return;
	}
	for (int i = ret; i < pc->sz - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("删除成功\n");

}

显示结果:
在这里插入图片描述

(7)按指定内容排序通讯录

使用qsort函数进行排序,排序的选项有按姓名排序和按年龄排序,排完序然后打印出来。

qsort函数原型:
void qsort (void* base, size_t num, size_t size,int (compar)(const void,const void*));
void* base – 待排序的数组的起始位置
size_t num – 元素的个数
size_t size – 一个元素的大小
int (compar)(const void,const void*) – 2个元素的比较函数

//清空所有联系人
void EmptyContact(Contact* pc)
{
	pc->sz = 0;
	printf("清空成功!\n");
}



//按姓名排序
int name_cmp(const void* e1, const void* e2)
{
	return strcmp(((PeopInfor*)e1)->name, ((PeopInfor*)e2)->name);
}
//按年龄排序
int age_cmp(const void* e1, const void* e2)
{
	return ((PeopInfor*)e1)->age - ((PeopInfor*)e2)->age;
}
//按指定内容排序所有联系人
void SortContact(Contact* pc)
{
	printf("*********    请选择排序方式(升序)    **********\n");
	printf("************ 1.姓名        2.年龄 *************\n");
	printf("**********************************************\n");
	int input = 0;
	scanf("%d", &input);
	switch (input)
	{
	case 1:
		qsort(pc->data, pc->sz, sizeof(pc->data[0]), name_cmp);
		break;
	case 2:
		qsort(pc->data, pc->sz, sizeof(pc->data[0]), age_cmp);
		break;
	default:
		printf("无此选项\n");
		break;
	}
	printf("%-5s\t%-4s\t%-5s\t%-15s%-20s\n",
		"姓名", "年龄", "性别", "电话", "地址");
	for (int i = 0; i < pc->sz; i++)
	{
		printf("%-5s\t%-4d\t%-5s\t%-15s%-20s\n",
			pc->data[i].name,
			pc->data[i].age,
			pc->data[i].sex,
			pc->data[i].tele,
			pc->data[i].addr);
	}
	printf("排序成功!\n");
}

显示结果:
在这里插入图片描述

(8)清空通讯录

把结构体中sz变为0就没有信息了

//清空所有联系人
void EmptyContact(Contact* pc)
{
	pc->sz = 0;
	printf("清空成功!\n");
}

显示结果:
在这里插入图片描述

(9)保存通讯录的信息

我们以只写wb打开一个二进制文件,把地址放在指针pf里,然后使用for循环把sz个信息写到pf里。

fwrite函数的原型:
size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream)
ptr-- 这是指向要被写入的元素数组的指针。
size-- 这是要被写入的每个元素的大小,以字节为单位。
nmemb-- 这是元素的个数,每个元素的大小为 size 字节。
stream-- 这是指向 FILE 对象的指针,该 FILE 对象指定了一个输出流。

//保存通讯录
void SaveContact(Contact* pc)
{
	FILE* pf = fopen("contact.txt", "wb");
	if (NULL == pf)
	{
		perror("SaveContact");
	}
	else
	{
		int i = 0;
		for (i = 0; i < pc->sz; i++)
		{
			fwrite(pc->data + i, sizeof(PeopInfor), 1, pf);
		}
		fclose(pf);
		pf = NULL;
		printf("保存成功!\n");
	}
}
(10)释放内存

通讯录中使用的是动态开辟的空间,是在堆区上面使用的,需要我们释放,使用free函数,一层一层释放。

void DestroyContact(Contact* pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->capacity = 0;
	pc->sz = 0;
	pc = NULL;
}

三、代码呈现
(1)test.c
#define _CRT_SECURE_NO_WARNINGS 1

#include"contact.h"

void menu()
{
	printf("*************************************\n");
	printf("******** 1.add      2.dele   ********\n");
	printf("******** 3.search   4.modify ********\n");
	printf("******** 5.show     6.empty  ********\n");
	printf("******** 7.sort     0.exit   ********\n");
	printf("*************************************\n");
}


enum Option
{
	EXIT,
	ADD,
	DELE,
	SEARCH,
	MODIFY,
	SHOW,
	EMPTY,
	SORT
};

int main()
{
	Contact con;
	InitContac(&con);
	int input = 0;
	do
	{
		menu();
		printf("请选择: ");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContacet(&con);
			break;
		case DELE:
			DeleContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case MODIFY:
			ModefyContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case EMPTY:
			EmptyContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case EXIT:
			SaveContact(&con);
			DestroyContact(&con);
			printf("退出\n");
			break;
		default:
			printf("没有此选项,请重新输入!\n");
			break;
		}

	} while (input);
	return 0;
}

(2)contact.c
#define _CRT_SECURE_NO_WARNINGS 1

#include"contact3.h"

通讯录初始化,静态版本
//void InitContac(Contact* pc)
//{
//	assert(pc);
//	pc->sz = 0;
//	memset(pc->data, 0, sizeof(pc->data));
//}

//通讯录初始化,动态版本
void InitContac(Contact* pc)
{
	assert(pc);
	pc->sz = 0;
	PeopInfor* ptr = (PeopInfor*)calloc(DEFAULT_SZ, sizeof(PeopInfor));
	if (ptr == NULL)
	{
		perror("InitContac::calloc");
		return;
	}
	pc->data = ptr;
	pc->capacity = DEFAULT_SZ;

	LodeContact(pc);
}

//销毁通讯录
void DestroyContact(Contact* pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->capacity = 0;
	pc->sz = 0;
	pc = NULL;
}

//添加通讯录人员,静态版本
//void AddContacet(Contact* pc)
//{
//	if (pc->sz == MAX)
//	{
//		printf("通讯录人员已满,无法添加\n");
//		return;
//	}
//	int num = 0;
//	printf("请输入要增加的人数:");
//	scanf("%d", &num);
//	int i = 0;
//	for (i = 0; i < num; i++)
//	{
//		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++;
//	}
//}


void Check_capacity(Contact* pc)
{
	if (pc->sz == pc->capacity)
	{
		PeopInfor* ptr = (PeopInfor*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeopInfor));
		if (ptr == NULL)
		{
			perror("Check_capacity::realloc");
			return;
		}
		pc->capacity += INC_SZ;
		pc->data = ptr;
		printf("增容成功\n");
	}
}

//添加通讯录人员,动态版本
void AddContacet(Contact* pc)
{
	assert(pc);
	Check_capacity(pc);
	int num = 0;
	printf("请输入要增加的人数:");
	scanf("%d", &num);
	int i = 0;
	for (i = 0; i < num; i++)
	{
		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);
		printf("添加成功!\n");
		pc->sz++;
	}
}


//显示通讯录名单
void ShowContact(const Contact* pc)
{
	assert(pc);
	printf("%-5s\t%-4s\t%-5s\t%-15s%-20s\n",
		"姓名", "年龄", "性别", "电话", "地址");
	for (int i = 0; i < pc->sz; i++)
	{
		printf("%-5s\t%-4d\t%-5s\t%-15s%-20s\n",
			pc->data[i].name,
			pc->data[i].age,
			pc->data[i].sex,
			pc->data[i].tele,
			pc->data[i].addr);
	}
}


int FindByName(const Contact* pc, char name[])
{
	assert(pc);
	for (int pos = 0; pos < pc->sz; pos++)
	{
		if (strcmp(pc->data[pos].name, name) == 0)
		{
			return pos;
		}
	}
	return -1;
}

//删除指定联系人
void DeleContact(Contact* pc)
{
	char name[NAME_MAX];
	assert(pc);
	printf("请输入要删除人的姓名:");
	scanf("%s", name);
	if (pc->sz == 0)
	{
		printf("通讯录为空,无法删除\n");
	}
	int ret = FindByName(pc, name);
	if (-1 == ret)
	{
		printf("要删除的人不存在\n");
		return;
	}
	for (int i = ret; i < pc->sz - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("删除成功\n");

}

//查找指定联系人
void SearchContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX];
	printf("请输入要查找人的姓名:");
	scanf("%s", name);
	int ret = FindByName(pc, name);
	if (-1 == ret)
	{
		printf("要查找的人不存在\n");
		return;
	}

	printf("%-5s\t%-4s\t%-5s\t%-15s%-20s\n",
		"姓名", "年龄", "性别", "电话", "地址");
		printf("%-5s\t%-4d\t%-5s\t%-15s%-20s\n",
			pc->data[ret].name,
			pc->data[ret].age,
			pc->data[ret].sex,
			pc->data[ret].tele,
			pc->data[ret].addr);
}


enum Option
{
	EXIT,
	NAME,
	AGE,
	SEX,
	TEL,
	ADDR,
	TOTAL
};

//修改指定联系人
void ModefyContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX];
	printf("请输入要修改人的姓名:");
	scanf("%s", name);
	int ret = FindByName(pc, name);
	if (-1 == ret)
	{
		printf("要修改的人不存在\n");
		return;
	}
	else
	{
		int num = 0;
		do
		{
			printf("请输入你想修改此联系人的信息:\n");
			printf("0.退出  1.姓名  2.年龄  3.性别  4.电话  5.地址  6.全部修改\n");
			scanf("%d", &num);
			switch (num)
			{
			case NAME:
				printf("请输入名字:");
				scanf("%s", pc->data[ret].name);
				printf("修改成功!\n");
				break;
			case AGE:
				printf("请输入年龄:");
				scanf("%d", &(pc->data[ret].age));
				printf("修改成功!\n");
				break;
			case SEX:
				printf("请输入性别:");
				scanf("%s", pc->data[ret].sex);
				printf("修改成功!\n");
				break;
			case TEL:
				printf("请输入电话:");
				scanf("%s", pc->data[ret].tele);
				printf("修改成功!\n");
				break;
			case ADDR:
				printf("请输入地址:");
				scanf("%s", pc->data[ret].addr);
				printf("修改成功!\n");
				break;
			case TOTAL:
				printf("请输入名字:");
				scanf("%s", pc->data[ret].name);
				printf("请输入年龄:");
				scanf("%d", &(pc->data[ret].age));
				printf("请输入性别:");
				scanf("%s", pc->data[ret].sex);
				printf("请输入电话:");
				scanf("%s", pc->data[ret].tele);
				printf("请输入地址:");
				scanf("%s", pc->data[ret].addr);
				printf("修改成功!\n");
				break;
			case EXIT:
				printf("退出\n");
				break;
			default:
				printf("无此选项\n");
				break;
			}
		} while (num);
	}
}


//清空所有联系人
void EmptyContact(Contact* pc)
{
	pc->sz = 0;
	printf("清空成功!\n");
}



//按姓名排序
int name_cmp(const void* e1, const void* e2)
{
	return strcmp(((PeopInfor*)e1)->name, ((PeopInfor*)e2)->name);
}
//按年龄排序
int age_cmp(const void* e1, const void* e2)
{
	return ((PeopInfor*)e1)->age - ((PeopInfor*)e2)->age;
}
//按指定内容排序所有联系人
void SortContact(Contact* pc)
{
	printf("*********    请选择排序方式(升序)    **********\n");
	printf("************ 1.姓名        2.年龄 *************\n");
	printf("**********************************************\n");
	int input = 0;
	scanf("%d", &input);
	switch (input)
	{
	case 1:
		qsort(pc->data, pc->sz, sizeof(pc->data[0]), name_cmp);
		break;
	case 2:
		qsort(pc->data, pc->sz, sizeof(pc->data[0]), age_cmp);
		break;
	default:
		printf("无此选项\n");
		break;
	}
	printf("%-5s\t%-4s\t%-5s\t%-15s%-20s\n",
		"姓名", "年龄", "性别", "电话", "地址");
	for (int i = 0; i < pc->sz; i++)
	{
		printf("%-5s\t%-4d\t%-5s\t%-15s%-20s\n",
			pc->data[i].name,
			pc->data[i].age,
			pc->data[i].sex,
			pc->data[i].tele,
			pc->data[i].addr);
	}
	printf("排序成功!\n");
}


//保存通讯录
void SaveContact(Contact* pc)
{
	FILE* pf = fopen("contact.txt", "wb");
	if (NULL == pf)
	{
		perror("SaveContact");
	}
	else
	{
		int i = 0;
		for (i = 0; i < pc->sz; i++)
		{
			fwrite(pc->data + i, sizeof(PeopInfor), 1, pf);
		}
		fclose(pf);
		pf = NULL;
		printf("保存成功!\n");
	}
}


//加载联系人
void LodeContact(Contact* pc)
{
	FILE* pf = fopen("contact.txt", "rb");
	if (NULL == pf)
	{
		perror("LodeContact");
	}
	else
	{
		PeopInfor temp = { 0 };
		int i = 0;
		while (fread(&temp, sizeof(PeopInfor), 1, pf))
		{
			Check_capacity(pc);
			pc->data[i] = temp;
			pc->sz++;
			i++;
		}
		fclose(pf);
		pf = NULL;
	}
}
(3)contact.h
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<string.h>


#define NAME_MAX 10
#define SEX_MAX 4
#define TELE_MAX 12
#define ADDR_MAX 20

#define DEFAULT_SZ 3
#define INC_SZ 2

typedef struct PeopInfor
{
	char name[NAME_MAX];
	int age;
	char sex[SEX_MAX];
	char tele[TELE_MAX];
	char addr[ADDR_MAX];
}PeopInfor;


静态版本
//typedef struct Contact
//{
//	PeopInfor data[MAX];
//	int sz;
//}Contact;


//动态版本
typedef struct Contact
{
	PeopInfor* data;//通讯录人员的信息
	int sz;//记录通讯录有多少人的信息
	int capacity;//记录当前通讯录的总容量
}Contact;


//通讯录初始化
void InitContac(Contact* pc);
//销毁通讯录
void DestroyContact(Contact* pc);
//添加通讯录人员
void AddContacet(Contact* pc);
//显示通讯录名单
void ShowContact(const Contact* pc);
//删除指定联系人
void DeleContact(Contact* pc);
//查找指定联系人
void SearchContact(Contact* pc);
//修改指定联系人
void ModefyContact(Contact* pc);
//清空所有联系人
void EmptyContact(Contact* pc);
//按姓名排序所有联系人
void SortContact(Contact* pc);
//保存通讯录
void SaveContact(Contact* pc);
//加载联系人
void LodeContact(Contact* pc);

看到这了,如果你似懂非懂就在此留下你阅读的痕迹吧😎

  • 9
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
一:需求分析 二:总体设计 三:详细设计 四:调试与测试 五;测试结果 六:本次课程设计的心得体会 七:附录 本程序设计亮点介绍(及)改进方面 1.在修改函数中,可以实现单条记录修改(运用switch) 2.查询函数中,能返回该函数继续查询(运用了递归) 3.使用变量少,仅通过变量i的递加和表示成员数的变量n就实现了数据结构体之间的联 系 4.界面美观,使用方便 5.允许编号输入为字符 6.在显示函数上能够分屏显示(换页) 缺点 在输入数据时,未输完一个成员的所有信息不能退出输入 一:需求分析 A.需要实现的功能有:输入功能、输出功能、显示功能、查找功能、以及增加、删 除、修改功能。输入功能要求一次完成若干信息的输入;显示功能要能完成所有信息的 显示,并且自动分页;查找可以通过多种方式进行。 按需求的功能,应该设计出输入函数、输出函数、显示函数、查找函数、修改函数。另 需要一个标准界面。 B.数据存储问题 这个系统数据庞大复杂,需要需要适当应用数组,循环,结构体,文件操作等基本 语句及稍有难度的语句。因此要求我们熟练掌握结构化程序设计的基本思路和方法,在 所掌握的基本知识和技能的基础上,进一步提高自学能力和查找资料的能力,解决一定 程度的复杂的结构化程序设计问题,加深对所学知识的理解与掌握,利用自己所学知识 解决实际问题的能力,为以后的程序开发打下基础。 二:总体设计 主要结构图示 本程序由主函数和多个子函数组成,可执行多种数据管理的功能。具体通过c语言中 结构体数组来实现。主函数主体为定义的结构体,和一个switch函数,以提供多种程序 操作功能。 主要需运用到结构体数组、文件、函数等内容知识。 子函数有六个,分别负责添加、显示、删除、查询、修改、保存的功能。通过变量i使结 构体数组之间达成数据链表的功能,利用该方法主要考虑到成员中对于结构体指针尚未 熟练掌握,而这种方法更加便捷易懂。 子函数功能介绍 1. 添加:输入成员信息,依次输入编号、姓名、QQ、电话号码、单位、电子邮箱、身 份。 2. 显示:以表格形式输出所有通讯记录 3. 删除:只需输入该成员姓名,便删除其所有记录 4. 查询:可选用姓名、电话、单位身份4种方式来查询 5. 修改:输入该成员姓名后,再依次输入更新信息完成修改 6. 保存:输入文件名(带后缀)后保存。C-free中会默认以txt文件保存至c- free standard/temp文件夹中 三:详细设计 1. 主函数 完成变量的初始化与函数的定义之后,运用for( ; ; )来开启一个无限循环。循环内有switch函数来实现函数的选择功能,case1、case2等分 别执行不同的函数操作,每个函数中调用了相应的函数,并采用函数system("pause") ;暂停等待按键。在case6中采用函数exit(0)跳出循环。 下面在讲一下主函数中定义的n的作用,n的初值是1,当执行完case1中的函数input ( )后返回值为n+i,i为输入的成员数,返回值重新赋给n,n变为成员数,从而确定了之后 display()等函数循环中的终止条件。 2. 分函数 【1】添加 【2】显示 【3】删除 【4】查询 【5】修改 保存函数:【1】定义一个文件指针fp 【2】输入想创建的文件名称,如filename 【3】开辟文件,用fopen以"写"的方式打开,并把fp指针指向该文件 【4】在文件中输入信息 【5】保存(即fclose(filename)) 查询函数:运用了if来实现选择,并通过递归返回继续查询。 注:身份以及单位的查询方式有所不同,单位只能查找一条符合的记录,而按身份查询 可显示所有符合的信息。若查找不到,则返回继续查询,选择5返回则退出查询。 四:调试与测试 我们编写的函数具有一定的容错能力。具体表现为 【1】如select函数中输入0-6以外的数,则循环继续进行,直到输入0-6为止 【2】sign!='n'&&sign!='N'语句保证了大小写不会影响结果 【3】while(m!='1'&&m!='2'&&m!='3'&&m!='4') { printf("输入错误,请重新选择:"); scanf("%c",&m); }保证了输入正确 【4】当输入的成员信息错误时,可通过修改函数进行改动 五:测试结果 1.主函数运行正常。通过for语句是界面循环,并有switch及system函数进行配合调节 达到目的。(界面显示如下) 2.测试结果:运行正常,达到了预订的目标。 此处为通过保存函数保存后的txt文本中的学生信息 六:心得 课程设计是一个不断学习、思考、编写、修改的过程。它对我们的能力提出了挑战。 我不仅意识到c语言基础的重要性,更意识到了自学能力的重要性,同学学习书本,查找 资料等多种途径来构建思

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

川子767

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值