C语言小项目——通讯录高阶(文件管理版)

该文章展示了一个通讯录管理系统的实现,包括动态内存分配以适应联系人数量的增长,文件加载和保存功能,以及对联系人信息的添加、删除、查找、修改和排序操作。系统使用结构体数组存储联系人信息,并通过检查容量来确保空间足够。
摘要由CSDN通过智能技术生成

通讯录初阶: 点这里
通讯录中阶: 点这里

文件管理版本改进之处

通讯录初始化

contact.c
在这里插入图片描述

退出通讯录并保存

test.c
在这里插入图片描述

contact.c
在这里插入图片描述

contact.h
在这里插入图片描述

完整代码

contact.h

#pragma once

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

#define MAX_CONTACT 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
//动态版本
#define DEFAULT_SIZE 3
#define INC_SIZE 2

enum OPTION
{
	EXIT0,
	ADD,
	DEL,
	SEARCH,
	MODIFY,
	SHOW,
	SORT
};

enum Peo
{
	EXIT1,
	NAME,
	AGE,
	SEX,
	TELE,
	ADDR
};

//一个人信息的结构体
typedef struct People
{
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
}Peo;

//存放大量人信息的通讯录
// 静态版本
//typedef struct Contact
//{
//	Peo data[MAX_CONTACT];
//	int sz;
//}Con;
//动态版本
//最初容量设置为3,当放满之后每次动态开辟2个容量
typedef struct Contact
{
	Peo* data;//指向存放数据的空间
	int sz;//记录当前通讯录有效元素的个数
	int capacity;//通讯录当前最大容量
}Con;

//初始化通讯录
void InitContact(Con* pc);
//给通讯录添加联系人
void AddContact(Con* pc);
//显示通讯录信息
void ShowContact(const Con* pc);
//删除指定联系人
void DelContact(Con* pc);
//查找指定联系人
void SearchContact(const Con* pc);
//修改指定联系人的信息
void ModifyContact(Con* pc);
//按照指定方式排序
void SortContact(Con* pc);
//文件版本
//把通讯录保存到文件中
void SaveContact(Con* pc);
//动态版本
//销毁通讯录
void DestroyContact(Con* pc);

contact.c

#define _CRT_SECURE_NO_WARNINGS 1

#include"contact.h"

//初始化通讯录——————————————————————————————————————————————————————————
//静态版本
//void InitContact(Con* pc)
//{
//	assert(pc);
//
//	//循环初始化也可
//	//int i = 0;
//	//for (i = 0; i < MAX_CONTACT; i++)
//	//{
//	//	strcpy(pc->data[i].name, "0");
//	//	pc->data[i].age = 0;
//	//	strcpy(pc->data[i].sex, "0");
//	//	strcpy(pc->data[i].tele, "0");
//	//	strcpy(pc->data[i].addr, "0");
//	//}
//	//pc->sz = 0;
//
//	//初始化结构体数组最简单的方法
//	memset(pc->data, 0, sizeof(pc->data));
//	pc->sz = 0;
//}

//动态版本
//void InitContact(Con* pc)
//{
//	assert(pc);
//
//	pc->data = (Peo*)malloc(DEFAULT_SIZE * sizeof(Peo));
//	if (pc->data == NULL)
//	{
//		perror(InitContact);
//		return;
//	}
//	pc->sz = 0;
//	pc->capacity = DEFAULT_SIZE;
//}

//文件版本
static int CheckCapacity(Con* pc)
{
	if (pc->sz == pc->capacity)
	{
		Peo* ptr = realloc(pc->data, (pc->capacity + INC_SIZE) * sizeof(Peo));
		if (ptr == NULL)
		{
			perror("CheakCapacity");
			return 0;
		}
		else
		{
			pc->data = ptr;
			pc->capacity += INC_SIZE;
			return 1;
		}
	}
	return 1;
}

static void LoadContact(Con* pc)
{
	//打开文件
	FILE* pf = fopen("contact.dat", "rb");
	if (pf == NULL)
	{
		perror("LoadContact");
		return;
	}
	//读文件
	Peo tmp = { 0 };
	while (fread(&tmp, sizeof(Peo), 1, pf))
	{
		if (0 == CheckCapacity(pc))
		{
			return;
		}
		pc->data[pc->sz] = tmp;
		pc->sz++;
	}
	//关闭文件
	fclose(pf);
	pf = NULL;
}

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

	pc->data = (Peo*)malloc(DEFAULT_SIZE * sizeof(Peo));
	if (pc->data == NULL)
	{
		perror(InitContact);
		return;
	}
	pc->sz = 0;
	pc->capacity = DEFAULT_SIZE;

	//文件中保存的信息加载到通讯录中
	LoadContact(pc);
}

//给通讯录添加联系人——————————————————————————————————————————————————————
//void AddContact(Con* pc)
//{
//	assert(pc);
//
//	if (pc->sz == MAX_CONTACT)
//	{
//		printf("通讯录已满,无法添加\n");
//		return;
//	}
//
//	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++;
//	printf("增加联系人成功\n");
//}

//动态版本
void AddContact(Con* pc)
{
	assert(pc);

	if (0 == CheckCapacity(pc))
	{
		return;
	}

	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++;
	printf("增加联系人成功\n");
}

//显示通讯录信息——————————————————————————————————————————————————————————
void ShowContact(const Con* pc)
{
	assert(pc);

	printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
	int i = 0;//如果后面需要i的值,就不能定义在for循环的初始化部分,因为出了循环就被销毁
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-20s\t%-4d\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(const Con* pc, char* name)
{
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(pc->data[i].name, name) == 0)
		{
			return i;
		}
	}
	return -1;
}

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

	if (pc->sz == 0)
	{
		printf("通讯录为空\n");
		return;
	}

	char name[MAX_NAME] = { 0 };
	printf("请输入要删除的人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要删除的人不存在\n");
		return;
	}
	//删除方法一:从后向前一个个覆盖
	//for (i = pos; i < pc->sz - 1; i++)
	//{
	//	pc->data[i] = pc->data[i + 1];
	//}
	//pc->sz--;

	//删除方法二:memmove,和方法一相同
	//memmove(&(pc->data[pos]), &(pc->data[pos + 1]), ((pc->sz) - pos - 1) * (sizeof(pc->data[0])));
	//pc->sz--;

	//删除方法三:将要删除的和最后一个交换,然后sz--
	Peo tmp = pc->data[pos];
	pc->data[pos] = pc->data[pc->sz - 1];
	pc->data[pc->sz - 1] = tmp;
	pc->sz--;

	printf("删除联系人成功\n");
}

//查找指定联系人————————————————————————————————————————————————————————————
void SearchContact(const Con* pc)
{
	assert(pc);

	//学习c++之后可以用函数重载实现用任何信息都能进行检索
	char name[MAX_NAME] = { 0 };
	printf("请输入要查找人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要查找的人不存在\n");
	}
	else
	{
		printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
		printf("%-20s\t%-4d\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(Con* pc)
{
	assert(pc);

	char name[MAX_NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要修改的人不存在");
	}
	else
	{
		//一股脑修改全部信息
		//printf("请输入名字:>");
		//scanf("%s", pc->data[pos].name);
		//printf("请输入年龄:>");
		//scanf("%d", &(pc->data[pos].age));
		//printf("请输入性别:>");
		//scanf("%s", pc->data[pos].sex);
		//printf("请输入电话:>");
		//scanf("%s", pc->data[pos].tele);
		//printf("请输入地址:>");
		//scanf("%s", pc->data[pos].addr);
		//printf("修改成功\n");

		//结合switch指定修改某个信息
		int input = 0;
		do
		{
			printf("0.EXIT1 1.NAME 2.AGE 3.SEX 4.TELE 5.ADDR\n");
			printf("请选择要修改的信息or选择0退出修改:>");
			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 EXIT1:
				break;
			default:
				printf("选择错误,重新选择\n");
				break;
			}
		} while (input);

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

//按照指定方式排序——————————————————————————————————————————————————————————————

int flag = 0;//qsort和Cmp函数已经固定好参数,要想体现升序和降序,只能定义全局变量,然后在函数里调用

static int CmpCharArr(const void* p1, const void* p2)
{
	return flag * (strcmp((*(Peo*)p1).name, (*(Peo*)p2).name));
}

static int CmpInt(const void* p1, const void* p2)
{
	return flag * (((*(Peo*)p1).age) - ((*(Peo*)p2).age));
}

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

	int input = 0;
	do
	{
		printf("0.EXIT1 1.NAME 2.AGE 3.SEX 4.TELE 5.ADDR\n");
		printf("请选择要按照哪种方式排序or选择0退出排序:>");
		scanf("%d", &input);
		if (input != 0)
		{
			printf("升序选择1,降序选择-1:>");
			while (flag != 1 && flag != -1)
			{
				scanf("%d", &flag);
				if (flag != 1 && flag != -1)
				{
					printf("输入错误,请重新输入\n");
				}
			}
		}
		switch (input)
		{
		case NAME:
			qsort(pc, pc->sz, sizeof(pc->data[0]), CmpCharArr);
			printf("排序成功\n");
			break;
		case AGE:
			qsort(pc, pc->sz, sizeof(pc->data[0]), CmpInt);
			printf("排序成功\n");
			break;
		case SEX:
			qsort(pc, pc->sz, sizeof(pc->data[0]), CmpCharArr);
			printf("排序成功\n");
			break;
		case TELE:
			qsort(pc, pc->sz, sizeof(pc->data[0]), CmpCharArr);
			printf("排序成功\n");
			break;
		case ADDR:
			qsort(pc, pc->sz, sizeof(pc->data[0]), CmpCharArr);
			printf("排序成功\n");
			break;
		case EXIT1:
			break;
		default:
			printf("选择错误,重新选择\n");
			break;
		}
	} while (input);
}

//文件版本
void SaveContact(Con* pc)
{
	FILE* pf = fopen("contact.dat", "wb");
	if (pf == NULL)
	{
		perror("SaveContact");
		return;
	}
	//写数据
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		fwrite(pc->data + i, sizeof(Peo), 1, pf);
	}

	//关闭文件
	fclose(pf);
	pf = NULL;
}

//动态版本
void DestroyContact(Con* pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->capacity = pc->sz = 0;
}

test.c

#define _CRT_SECURE_NO_WARNINGS 1

#include"contact.h"

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

void test()
{
	int input = 0;
	Con con;
	InitContact(&con);

	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:
			//c++要是有函数重载会好写很多
			SortContact(&con);
			break;
		case EXIT0:
			//文件版本
			SaveContact(&con);
			//动态版本
			DestroyContact(&con);
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误,重新选择\n");
			break;
		}
	} while (input);
}

int main()
{
	test();
	return 0;
}


  • 6
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值