通讯录(动态实现与文件优化版)

实现一个通讯录:

通讯录可以用来存储1000个人的信息,每个人的信息包括:姓名、性别、年龄、电话、住址
提供方法:
        添加联系人信息;
        删除指定联系人信息;
        查找指定联系人信息;
        修改指定联系人信息;
        显示所有联系人信息;
        清空所有联系人;
        以名字排序所有联系人;

第一步:

分别创建test.c ,tool.c ,too1.h 三个文件分别用来保存界面实现,游戏实现,函数声明。

第二步:test.c(源文件)

在源文件的主函数中创建menu(菜单),告诉玩家怎么操作!

并利用do-while语句,进入操作界面,通过scanf来实现操作输入,显示。

用户根据需求进行不同数字输入来进行不同操作。

自此,源文件基本操作实现完成。

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1


#include "test.h"

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


void menu()
{
	printf("****1.添加******\n");
	printf("****2.删除******\n");
	printf("****3.查找******\n");
	printf("****4.修改******\n");
	printf("****5.显示******\n");
	printf("****6.排序******\n");
	printf("****7.清空******\n");
	printf("****0.退出******\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 DELE:
			DELEContact(&con);
			break;
		case SEARCH:
			SEARCHContact(&con);
			break;
		case MODIFY:
			MODIFYContact(&con);
			break;
		case SHOW:
			SHOWContact(&con);
			break;
		case SORT:
			SORTContact(&con);
			break;
		case EXIT:
			SAVEContact(&con);
			DestroyContact(&con);
			printf("退出通讯录成功!\n");
			break;
		case CLEAR:
			CLEARContact(&con);
			printf("清空成功!\n");
			break;
		default :
			printf("输入格式错误,请重新输入!\n");
			break;
		}
	} while (input);
	return 0;
}

 第三步:tool.h(通讯录函数声明)

        创建通讯录的结构体,通过这个结构体去调用存储这些信息的结构体;完成在tool.h中增加联系人都通讯录ADDContact(),删除指定联系人DELEContact(),查找指定联系人SEARCHContact(),修改指定联系人MODIFYContact(),打印通讯中的信息SHOWContact(),按照名字来排序SORTContact(),保存通讯录的信息到文件SAVEContact(),销毁通讯录CLEARContact(),加载文件的信息到通讯录LoadContact(),清空通讯录CLEARContact()的函数声明。

#pragma once
#include <stdio.h>
#include <assert.h>
#include <string.h>
#include <stdlib.h>


//扩容时的扩容值
#define DEFAULT_SZ 3
#define INC_SZ 2


#define MAX 100
#define MAX_NAME  20
#define MAX_SEX 10
#define MAX_TELE 12
#define MAX_ADDR 30


//
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;


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

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

//增加联系人都通讯录
void ADDContact(Contact* pc);

//打印通讯中的信息
void SHOWContact(const Contact* pc);

//删除指定联系人
void DELEContact(Contact* pc);

//查找指定联系人
void SEARCHContact(Contact* pc);

//修改指定联系人
void MODIFYContact(Contact* pc);

//排序通讯录中内容
//按照名字来排序
void SORTContact(Contact* pc);

//保存通讯录的信息到文件
void SAVEContact(const Contact* pc);

//加载文件的信息到通讯录
void LoadContact(Contact* pc);

//清空通讯录
void CLEARContact(Contact* pc);

第四步:tool.c(通讯录实现)

        完成在tool.h中初始化通讯录InitContact(),增加联系人都通讯录ADDContact(),删除指定联系人DELEContact(),查找指定联系人SEARCHContact(),修改指定联系人MODIFYContact(),打印通讯中的信息SHOWContact(),按照名字来排序SORTContact(),保存通讯录的信息到文件SAVEContact(),销毁通讯录DestroyContact(),加载文件的信息到通讯录LoadContact(),清空通讯录CLEARContact()的实现。

首先:

        对通讯录初始化InitContac),开辟(PeoInfo*)类型的动态内存,并设置动态容量为DEFAULT_SZ然后再加载之前的通讯录信息,如果开辟的内存不够,再动态多开辟INC_SZ个内存空间

#include "test.h"

//初始化通讯录
//void InitContact(Contact* pc)
//{
//	assert(pc);
//	pc->count = 0;
//	memset(pc->data,0,sizeof(pc->data));
//}

void CheckCapacity(Contact* 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;
		}
		else
		{
			pc->data = ptr;
			pc->capacity += INC_SZ;
			printf("增容成功\n");
		}
	}
}

//动态的版本
int 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 1;
	}
	
	pc->capacity = DEFAULT_SZ;
	//加载文件的信息到通讯录中
	LoadContact(pc);

	return 0;
}

加载文件的信息到通讯录:


void LoadContact(Contact* pc)
{
	FILE* pfRead = fopen("contact.txt", "rb");
	if (pfRead == NULL)
	{
		perror("LoadContact");
		return;
	}

	PeoInfo tmp = { 0 };
	while (fread(&tmp, sizeof(PeoInfo), 1, pfRead) == 1)
	{
		CheckCapacity(pc);

		pc->data[pc->count] = tmp;
		pc->count++;
	}
	fclose(pfRead);
	pfRead = NULL;
}

 销毁通讯录:

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

增加联系人都通讯录:

//动态的版本
void ADDContact(Contact* pc)
{
	assert(pc);
	// 增容
	CheckCapacity(pc);
	/*if (pc->count == MAX)
	{
		printf("通讯录已满!\n");
		return;
	}*/

	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].tele);
	printf("请输入地址:\n");
	scanf("%s",pc->data[pc->count].addr);

	pc->count++;
	printf("添加成功!\n");
}

打印通讯中的信息:

void SHOWContact(const Contact* pc)
{
	assert(pc);
	int i = 0;
	printf("%-20s\t%-5s\t%-5s\t%-12s\t%-20s\n","姓名","年龄","性别","电话","地址");
	for (i = 0;i < pc->count;i++)
	{
		printf("%-20s\t%-5d\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);
	}

}

删除指定联系人:

static int FindByName(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 DELEContact(Contact* pc)
{
	assert(pc);
	int i = 0;
	char name[MAX_NAME] = { 0 };
	if (pc->count == 0)
	{
		printf("通讯录为空,没有可删除的信息!\n");
		return;
	}
	printf("请输入要删除的人的名字:\n");
	scanf("%s", name);
	//查找
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("需要删除的人不存在!\n");
		return;
	}
	//删除
	for (i = pos; i < pc->count - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->count--;
	printf("删除成功!\n");

}

查找指定联系人:

void SEARCHContact(Contact* pc)
{
	assert(pc);
	char name[MAX_NAME] = { 0 };
	printf("请输入要查询的人的名字:\n");
	scanf("%s",name);
	int pos = FindByName(pc,name);
	if (pos == -1)
	{
		printf("没有查找到此人:\n");
		return;
	}
	//显示
	printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	printf("%-20s\t%-5d\t%-5s\t%-12s\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);
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改信息的人的名字:\n");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("没有找到需要修改信息的人:\n");
		return;
	}
	printf("已找到需要修改信息的人,请修改:\n");
	//修改
	printf("请输入名字:\n");
	scanf("%s",pc->data[pos].name);
	printf("请输入年龄:\n");
	scanf("%d",&(pc->data[pos].age));
	printf("请输入性别:\n");
	scanf("%s",pc->data[pos].sex);
	printf("请输入电话:\n");
	scanf("%s",pc->data[pos].tele);
	printf("请输入地址:\n");
	scanf("%s",pc->data[pos].addr);

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

按照名字来排序:

int cmp_Peo_age(const void* p1, const void* p2)
{
	return strcmp(((PeoInfo*)p1)->name, ((PeoInfo*)p2)->name);
}


//按照名字来排序
void SORTContact(Contact* pc)
{
	assert(pc);
	qsort(pc->data,pc->count,sizeof(PeoInfo),cmp_Peo_age);
	printf("排序完成!\n");
}

保存通讯录的信息到文件:

void SAVEContact(const Contact* pc)
{
	assert(pc);
	FILE* pfWrite = fopen("contact.txt", "wb");
	if (pfWrite == NULL)
	{
		perror("SAVEContact");
		return;
	}
	//写文件-二进制的形式
	int i = 0;
	for (i = 0; i < pc->count; i++)
	{
		fwrite(pc->data+i,sizeof(PeoInfo),1,pfWrite);
	}
	fclose(pfWrite);
	pfWrite = NULL;
}

清空通讯录

        在退出时,保存通讯录中的信息,再释放动态开辟的内存

//清空
void CLEARContact(Contact* pc)
{
	assert(pc);
	pc->count = 0;
	pc->capacity = 0;
	memset(pc->data,0,sizeof(pc->data));

}

如下为tool.c的全部实现集合

#define _CRT_SECURE_NO_WARNINGS 1

#include "test.h"



//初始化通讯录
//void InitContact(Contact* pc)
//{
//	assert(pc);
//	pc->count = 0;
//	memset(pc->data,0,sizeof(pc->data));
//}


void CheckCapacity(Contact* 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;
		}
		else
		{
			pc->data = ptr;
			pc->capacity += INC_SZ;
			printf("增容成功\n");
		}
	}
}




void LoadContact(Contact* pc)
{
	FILE* pfRead = fopen("contact.txt", "rb");
	if (pfRead == NULL)
	{
		perror("LoadContact");
		return;
	}

	PeoInfo tmp = { 0 };
	while (fread(&tmp, sizeof(PeoInfo), 1, pfRead) == 1)
	{
		CheckCapacity(pc);

		pc->data[pc->count] = tmp;
		pc->count++;
	}
	fclose(pfRead);
	pfRead = NULL;
}




//动态的版本
int 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 1;
	}
	
	pc->capacity = DEFAULT_SZ;
	//加载文件的信息到通讯录中
	LoadContact(pc);

	return 0;
}






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


//动态的版本
void ADDContact(Contact* pc)
{
	assert(pc);
	// 增容
	CheckCapacity(pc);
	/*if (pc->count == MAX)
	{
		printf("通讯录已满!\n");
		return;
	}*/

	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].tele);
	printf("请输入地址:\n");
	scanf("%s",pc->data[pc->count].addr);

	pc->count++;
	printf("添加成功!\n");
}




void SHOWContact(const Contact* pc)
{
	assert(pc);
	int i = 0;
	printf("%-20s\t%-5s\t%-5s\t%-12s\t%-20s\n","姓名","年龄","性别","电话","地址");
	for (i = 0;i < pc->count;i++)
	{
		printf("%-20s\t%-5d\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);
	}

}


static int FindByName(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 DELEContact(Contact* pc)
{
	assert(pc);
	int i = 0;
	char name[MAX_NAME] = { 0 };
	if (pc->count == 0)
	{
		printf("通讯录为空,没有可删除的信息!\n");
		return;
	}
	printf("请输入要删除的人的名字:\n");
	scanf("%s", name);
	//查找
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("需要删除的人不存在!\n");
		return;
	}
	//删除
	for (i = pos; i < pc->count - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->count--;
	printf("删除成功!\n");

}




void SEARCHContact(Contact* pc)
{
	assert(pc);
	char name[MAX_NAME] = { 0 };
	printf("请输入要查询的人的名字:\n");
	scanf("%s",name);
	int pos = FindByName(pc,name);
	if (pos == -1)
	{
		printf("没有查找到此人:\n");
		return;
	}
	//显示
	printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	printf("%-20s\t%-5d\t%-5s\t%-12s\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);
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改信息的人的名字:\n");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("没有找到需要修改信息的人:\n");
		return;
	}
	printf("已找到需要修改信息的人,请修改:\n");
	//修改
	printf("请输入名字:\n");
	scanf("%s",pc->data[pos].name);
	printf("请输入年龄:\n");
	scanf("%d",&(pc->data[pos].age));
	printf("请输入性别:\n");
	scanf("%s",pc->data[pos].sex);
	printf("请输入电话:\n");
	scanf("%s",pc->data[pos].tele);
	printf("请输入地址:\n");
	scanf("%s",pc->data[pos].addr);

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



int cmp_Peo_age(const void* p1, const void* p2)
{
	return strcmp(((PeoInfo*)p1)->name, ((PeoInfo*)p2)->name);
}


//按照名字来排序
void SORTContact(Contact* pc)
{
	assert(pc);
	qsort(pc->data,pc->count,sizeof(PeoInfo),cmp_Peo_age);
	printf("排序完成!\n");
}



void SAVEContact(const Contact* pc)
{
	assert(pc);
	FILE* pfWrite = fopen("contact.txt", "wb");
	if (pfWrite == NULL)
	{
		perror("SAVEContact");
		return;
	}
	//写文件-二进制的形式
	int i = 0;
	for (i = 0; i < pc->count; i++)
	{
		fwrite(pc->data+i,sizeof(PeoInfo),1,pfWrite);
	}
	fclose(pfWrite);
	pfWrite = NULL;
}



//清空
void CLEARContact(Contact* pc)
{
	assert(pc);
	pc->count = 0;
	pc->capacity = 0;
	memset(pc->data,0,sizeof(pc->data));

}

以上为通讯录基础实现,可能有遗漏操作,敬请谅解!

猜你喜欢

转载自blog.csdn.net/weixin_71964780/article/details/132137839