【C语言】静态通讯录

今天我们利用所学的c语言知识做一个简易的静态通讯录,一起来看一下吧

目录

简易通讯录要求

人的信息处理

初始化通讯录函数

简易菜单&实现思路

增加联系人函数

显示通讯录函数

删除联系人函数

代码优化

查找联系人函数

修改联系人函数

排序通讯录函数

完整代码

contact.h

contact.c

test.c


简易通讯录要求

首先我们了解一下通讯录需要涵盖的内容有哪些:

1.人的信息:姓名,年龄,性别,地址,电话

2.通讯录中可以存放100个人的信息

3.功能:
1>增加联系人
2>删除指定联系人
3>查找指定联系人的信息
4>修改指定联系人的信息
5>显示所有联系人信息
6>排序(名字/年龄)

人的信息处理

首先呢针对人的信息,我们可以使用结构体来完成:

//人的信息
typedef struct PeoInfo {
	char name[20];
	int age;
	char sex[5];
	char addr[30];
	char tele[12];
}PeoInfo;

因为我们后续需要。存放100个人的信息,因此我们可以再创建一个结构体

typedef struct Contact {
	PeoInfo data[100];//存放人的信息
	int sz;//当前已经放入到信息个数
}Contact;

初始化通讯录函数

然后就是对数据进行初始化

void InitContact(Contact* pc) {
	pc->sz = 0;
	memset(pc->data, 0, sizeof(pc->data));
}

这里的memset函数就是为了把data数组内全部初始化为0

简易菜单&实现思路

为了更方便通讯录的实现,我们可以在test.c文件写一个简易的菜单,方便通讯录实现后对应操作

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

}

然后在main函数内部,我们可以用一个switch语句,将功能与菜单对应,然后在对应的case实现函数,switch语句之外,我们可以写一个do-while循环保证代码多次运行,这就是我们的简易思路,下面我们就来写对应的函数

增加联系人函数

首先是增加联系人AddContact函数:

void AddContact(Contact* pc) {
	if (pc->sz == MAX) {
		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].addr);
	printf("请输入电话:>");
	scanf("%s", pc->data[pc->sz].tele);

	pc->sz++;
}

显示通讯录函数

那么信息输入进去我们想看怎么办呢,那我们就先写ShowContact函数:

void ShowContact(Contact* pc) {
	printf("%-20s\t%-4s\t%-5s\t%-20s\t%-12s\n", "名字", "年龄", "性别", "地址", "电话");
	for (int i = 0; i < pc->sz; i++) {
		printf("%-20s\t%-4d\t%-5s\t%-20s\t%-12s\n", pc->data[i].name,
									   pc->data[i].age,
									   pc->data[i].sex,
									   pc->data[i].addr,
									   pc->data[i].tele);
	}
}

删除联系人函数

接下来我们写删除函数:

void DelContact(Contact* pc) {
	char name[NAME_MAX] = { 0 };
	if (pc->sz == 0) {
		printf("通讯录为空,无法删除\n");
		return;
	}
	//删除
	//找到要删除的人
	printf("请输入需要删除的人的名字:>");
	scanf("%s",name);
	int i = 0;
	int del = 0;
	for (i = 0; i > pc->sz; i++) {
		if (strcmp(pc->data[i].name, name) == 0) {
			del = i;
			break;
		}
	}
	//删除
	for (i = del; i < pc->sz-1; i++) {
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("删除成功\n");
}

代码优化

这里结束我们观察需要的函数不难发现,查找,删除,修改相关函数都有一个查找的过程,因此我们可以将删除函数中的查找元素的部分封装成一个函数如下:

int FindByName(Contact* 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(Contact* pc) {
	char name[NAME_MAX] = { 0 };
	if (pc->sz == 0) {
		printf("通讯录为空,无法删除\n");
		return;
	}
	//删除
	//找到要删除的人
	printf("请输入需要删除的人的名字:>");
	scanf("%s",name);
	int ret=FindByName(pc, name);
	if (-1 == ret) {
		printf("要删除的人不存在\n");
		return;
	}
	int i = 0;
	//删除
	for (i = ret; i < pc->sz-1; i++) {
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("删除成功\n");
}

查找联系人函数

接下来我们写查找函数:

void SearchContact(const Contact* pc) {
	char name[NAME_MAX] = { 0 };
	printf("请输入要查找的人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (-1 == pos) {
		printf("要查找的人不存在\n");
		return;
	}
	//打印信息
	printf("%-20s\t%-4s\t%-5s\t%-20s\t%-12s\n", "名字", "年龄", "性别", "地址", "电话");
	printf("%-20s\t%-4d\t%-5s\t%-20s\t%-12s\n", pc->data[pos].name,
		pc->data[pos].age,
		pc->data[pos].sex,
		pc->data[pos].addr,
		pc->data[pos].tele);

}

修改联系人函数

接下来是修改信息的函数,跟前面的函数有重叠的部分,可以复用:

void ModifyContact(Contact* pc) {
	char name[NAME_MAX] = { 0 };
	printf("请输入要修改的人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (-1 == pos) {
		printf("要修改的人不存在\n");
		return;
	}
	//修改
	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].addr);
	printf("请输入电话:>");
	scanf("%s", pc->data[pos].tele);
	printf("修改完成\n");
}

排序通讯录函数

最后的排序我们用qsort函数:

//排序
int cmp_con_by_name(const void* p1, const void* p2) {
	return(strcmp(((PeoInfo*)p1)->name, ((PeoInfo*)p2)->name));
}
void SortContact(const Contact* pc) {
	if (0 == pc->sz) {
		printf("通讯录为空,无法排序\n");
		return;
	}
	qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_con_by_name);
	printf("排序成功\n");

}

完整代码

至此,我们的静态通讯录就基本写完了下面附上全部代码

contact.h

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
#define MAX 100
#define NAME_MAX 20
#define SEX_MAX 5
#define ADDR_MAX 30
#define TELE_MAX 12

//人的信息
typedef struct PeoInfo {
	char name[NAME_MAX];
	int age;
	char sex[SEX_MAX];
	char addr[ADDR_MAX];
	char tele[TELE_MAX];
}PeoInfo;

typedef struct Contact {
	PeoInfo data[MAX];//存放人的信息
	int sz;//当前已经放入到信息个数
}Contact;

void InitContact(Contact* pc);//初始化通讯录
void AddContact(Contact* pc);//增加联系人
void ShowContact(const Contact* pc);//显示通讯录信息
void DelContact(Contact* pc);//删除通讯录信息
void SearchContact(const Contact* pc);//查找指定联系人
void ModifyContact(Contact* pc);//修改指定联系人
void SortContact(const Contact* pc);//按名字排序

contact.c

#define _CRT_SECURE_NO_WARNINGS
#include"contact.h"


void InitContact(Contact* pc) {
	pc->sz = 0;
	memset(pc->data, 0, sizeof(pc->data));
}

void AddContact(Contact* pc) {
	if (pc->sz == MAX) {
		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].addr);
	printf("请输入电话:>");
	scanf("%s", pc->data[pc->sz].tele);

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

int FindByName(const Contact* 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(Contact* pc) {
	char name[NAME_MAX] = { 0 };
	if (pc->sz == 0) {
		printf("通讯录为空,无法删除\n");
		return;
	}
	//删除
	//找到要删除的人
	printf("请输入需要删除的人的名字:>");
	scanf("%s",name);
	int ret=FindByName(pc, name);
	if (-1 == ret) {
		printf("要删除的人不存在\n");
		return;
	}
	int i = 0;
	//删除
	for (i = ret; i < pc->sz-1; i++) {
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("删除成功\n");
}

void SearchContact(const Contact* pc) {
	char name[NAME_MAX] = { 0 };
	printf("请输入要查找的人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (-1 == pos) {
		printf("要查找的人不存在\n");
		return;
	}
	//打印信息
	printf("%-20s\t%-4s\t%-5s\t%-20s\t%-12s\n", "名字", "年龄", "性别", "地址", "电话");
	printf("%-20s\t%-4d\t%-5s\t%-20s\t%-12s\n", pc->data[pos].name,
		pc->data[pos].age,
		pc->data[pos].sex,
		pc->data[pos].addr,
		pc->data[pos].tele);

}
void ModifyContact(Contact* pc) {
	char name[NAME_MAX] = { 0 };
	printf("请输入要修改的人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (-1 == pos) {
		printf("要修改的人不存在\n");
		return;
	}
	//修改
	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].addr);
	printf("请输入电话:>");
	scanf("%s", pc->data[pos].tele);
	printf("修改完成\n");
}
//排序
int cmp_con_by_name(const void* p1, const void* p2) {
	return(strcmp(((PeoInfo*)p1)->name, ((PeoInfo*)p2)->name));
}
void SortContact(const Contact* pc) {
	if (0 == pc->sz) {
		printf("通讯录为空,无法排序\n");
		return;
	}
	qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_con_by_name);
	printf("排序成功\n");

}

test.c

#define _CRT_SECURE_NO_WARNINGS
#include"contact.h"
//通讯录
//1.人的信息:姓名,年龄,性别,地址,电话
//2.通讯录中可以存放100个人的信息
//3.功能:
//1>增加联系人
//2>删除指定联系人
//3>查找指定联系人的信息
//4>修改指定联系人的信息
//5>显示所有联系人信息
//6>排序(名字/年龄)



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

}
int main() {
	int input = 0;
	//创建通讯录
	Contact con;
	//初始化通讯录
	InitContact(&con);

	do {
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input) {
		case 1:
			AddContact(&con);
			break;
		case 2:
			DelContact(&con);
			break;
		case 3:
			SearchContact(&con);
			break;
		case 4:
			ModifyContact(&con);
			break;
		case 5:
			ShowContact(&con);
			break;
		case 6:
			SortContact(&con);
			break;
		case 0:
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误\n");
			break;
		}
	} while (input);
	return 0;
}

我们容易发现这个通讯录最大的弊端就是空间问题,我们创建的通讯录大小是确定的,这就导致数据少浪费空间,数据过多空间又不够用,那么如何优化我们的静态通讯录呢,这就不得不提我们的动态通讯录了,如何实现呢?我们下一篇文章揭晓……

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值