今天我们利用所学的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");
}
至此,我们的静态通讯录就基本写完了下面附上全部代码
#define _CRT_SECURE_NO_WARNINGS
#include
#include
#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);//按名字排序
#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");
}
#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;
}
我们容易发现这个通讯录最大的弊端就是空间问题,我们创建的通讯录大小是确定的,这就导致数据少浪费空间,数据过多空间又不够用,那么如何优化我们的静态通讯录呢,这就不得不提我们的动态通讯录了,如何实现呢?我们下一篇文章揭晓……