文章目录
前言:对于整个通讯录的设计,我采用的是模块化设计,每一个功能,我通过封装一个个函数来实现,这对于开发人员来说,是非常友好的,也方便后期的调试。
通讯录
通讯录存储联系人的信息
通讯录可以用来存储1000个人的信息,每个人的信息包括:姓名、性别、年龄、电话、住址
存储联系人的信息,MAX_NAME 等这些我是用宏来定义的
我通过两个结构体
#define MAX 1000
#define MAX_NAME 20
#define SEX 10
#define TELE 12
#define ADDR 30
//人的信息
typedef struct PeoInfo
{
char name[MAX_NAME];
int age;
char sex[SEX];
char tele[TELE];
char addr[ADDR];
}PeoInfo;
//通讯录
typedef struct Contact
{
PeoInfo data[MAX];//只能保存1000个人的信息
int count; //记录当前通讯录里面保存联系人的个数
}Contact;
做完这些创建之后,还要结构体的初始化,
//Contact 初始化
void InitContact(Contact* pc)
{
assert(pc);
pc->count = 0;
memset(pc->data, 0, sizeof(pc->data));
}
对于初始化,我用到了一个memset 的库函数
添加联系人
添加联系人,就是把联系人的信息存储到data[MAX_NAME]的数组中,但是添加 完成的同时,我们需要把count+1,记录当前一共添加了多少个联系人。要会使用结构体指针
//Contact 增加通讯录联系人的信息
void AddContact(Contact* pc)
{
assert(pc);
if (pc->count == MAX)
{
printf("通讯录已满,无法添加!\n");
return;
}
else
{
printf("请输入姓名:>");
scanf("%s", pc->data[pc->count].name);//name 是数组名,是地址
printf("请输入年龄:>");
scanf("%d", &(pc->data[pc->count].age));
printf("请输入性别:>");
scanf("%s", pc->data[pc->count].sex);
printf("请输入电话:>");
scanf("%s", pc->data[pc->count].tele);
printf("请输入地址:>");
scanf("%s", pc->data[pc->count].addr);
pc->count++;
printf("增加成功!\n");
return;
}
}
删除指定联系人
想要删除联系人,首先你的判断当前通讯录里面是否存在这个联系人,找到了之后,就用后一个联系人覆盖当前这个,以此类推,和链表删除结点类似。
//删除通讯录的某一个联系人
void DelContact( Contact* pc)
{
assert(pc);
char name[MAX_NAME] = { 0 };
int i = 0;
//判断通讯录里是否有联系人
if (pc->count == 0)
{
printf("通讯录为空!\n");
return;
}
else
{
//找到通讯录里你要删除的人
printf("请输入你要删除联系人的姓名:>");
scanf("%s", name);
//查找
//封装一个函数,去查找,找到,返回下标,找不到返回-1
int ret = FindByName(pc, name);
if (ret == -1)
{
printf("你要删除的联系人不存在!\n");
return;
}
else
{
//pc->count - 1 的原因是:加入count = 100,
//则 i = 99时,i + 1 == 100,会造成指针越界
for (i = ret; i < pc->count - 1; i++)
{
//删除
pc->data[i] = pc->data[i + 1];
}
pc->count--;
}
}
printf("删除成功!\n");
}
查找指定联系人
查找指定联系人之前,先判断通讯录是否有联系人,我们这里是通过名字来查找的,但是这里有个弊端,就是通讯录出现同名的情况下,它会两个一起打印,这是新手项目,会存在一定的bug哦。
//查找,static 修饰的函数,只能在contact.c 文件中用,属于静态变量
static int FindByName(Contact* pc, const char* name)
{
assert(pc && name);
int i = 0;
for (i = 0; i < pc->count; i++)
{
if (0 == strcmp(pc->data[i].name, name))
{
return i;//找到
}
}
return -1;
}
//查找指定的联系人
void SearchContact(Contact* pc)
{
assert(pc);
char name[MAX_NAME] = { 0 };
int i = 0;
if (pc->count == 0)
{
printf("通讯录里没有联系人!\n");
return;
}
//查找
else
{
printf("请输入有查找的联系人:>");
scanf("%s", name);
//找到返回下标,不找到返回-1
int ret = FindByName(pc, name);
if (ret == -1)
{
printf("通讯录里不存在这个联系人!\n");
return;
}
else
{
//打印出来看一下
i = ret;
printf("%-20s\t%-10s\t%-10s\t%-12s\t%-30s\n",
"姓名", "年龄", "性别", "电话", "地址");
printf("%-20s\t%-10d\t%-10s\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);
}
printf("查找已完成!\n");
}
}
显示所有联系人信息
为了方便清楚的看到联系人的信息,我们先把菜单打印出来
//Contact 打印联系人的信息
void ShowContact( const Contact* pc)
{
assert(pc);//断言,防止pc是空指针
//判断通讯录中是否有联系人
if (pc->count == 0)
{
return;
}
else
{
printf("%-20s\t%-10s\t%-10s\t%-12s\t%-30s\n",
"姓名","年龄","性别","电话","地址");
int i = 0;
for(i = 0; i < pc->count; i++)
{
printf("%-20s\t%-10d\t%-10s\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);
}
printf("打印完成!\n");
return;
}
}
清空所有联系人
清空联系人是最简易实现的,我们只要把count 赋值成了0,就实现了
//清空所有联系人
void EmptyContact(Contact* pc)
{
//memset(pc->data, 0,sizeof(pc->data));
pc->count = 0;
printf("清空成功!\n");
return;
}
以名字排序所有联系人
/排序通讯录里面的联系人,按照名字来排
void SortContact(Contact* pc)
{
assert(pc);
if (pc->count == 0)
{
printf("通讯录是空的,没有联系人!\n");
return;
}
else
{
qsort(pc->data, pc->count, sizeof(PeoInfo), cmp_peo_by_name);
}
printf("排序成功!\n");
}
——————————————————————————————
项目的全部代码
通讯录测试代码
#define _CRT_SECURE_NO_WARNINGS
#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("***** 7. empty 0.exit *****\n");
printf("********************************\n");
}
enum dele
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SHOW,
SORT,
EMPTY
};
int main()
{
int input = 0;
Contact 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);//修改
ShowContact(&con);
break;
case SHOW:
ShowContact(&con);//打印联系人的信息
break;
case SORT:
SortContact(&con);//排序
ShowContact(&con);
break;
case EMPTY:
EmptyContact(&con);//清空所有联系人
break;
case EXIT:
printf("退出程序!\n");
break;
default :
printf("你的输入有误,请重新选择!\n");
break;
}
} while (input);
return 0;
}
通讯录所所有函数的声明和头文件
#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
#define MAX 1000
#define MAX_NAME 20
#define SEX 10
#define TELE 12
#define ADDR 30
//类型声明
//人的信息
typedef struct PeoInfo
{
char name[MAX_NAME];
int age;
char sex[SEX];
char tele[TELE];
char addr[ADDR];
}PeoInfo;
//通讯录
typedef struct Contact
{
PeoInfo data[MAX];//只能保存100个人的信息
int count; //记录当前通讯录里面保存联系人的个数
}Contact;
//Contact 初始化
void InitContact(Contact * pc);
//Contact 增加人的信息
void AddContact(Contact* pc);
//Contact 打印联系人的信息
void ShowContact(const Contact* pc);
//删除通讯录的某一个联系人
void DelContact(Contact* pc);
//查找指定的联系人
void SearchContact(Contact * pc);
//修改指定联系人的信息
void ModifyContact(Contact* pc);
//排序通讯录里面的联系人,按照名字来排
void SortContact(Contact* pc);
//清空所有联系人
void EmptyContact(Contact* pc);
通讯录函数的实现代码
#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"
//Contact 初始化
void InitContact(Contact* pc)
{
assert(pc);
pc->count = 0;
memset(pc->data, 0, sizeof(pc->data));
}
//Contact 增加通讯录联系人的信息
void AddContact(Contact* pc)
{
assert(pc);
if (pc->count == MAX)
{
printf("通讯录已满,无法添加!\n");
return;
}
else
{
//printf("%20s %3d %10s %12s %30s\n","姓名","年龄","性别","电话","地址");
printf("请输入姓名:>");
scanf("%s", pc->data[pc->count].name);//name 是数组名,是地址
printf("请输入年龄:>");
scanf("%d", &(pc->data[pc->count].age));
printf("请输入性别:>");
scanf("%s", pc->data[pc->count].sex);
printf("请输入电话:>");
scanf("%s", pc->data[pc->count].tele);
printf("请输入地址:>");
scanf("%s", pc->data[pc->count].addr);
pc->count++;
printf("增加成功!\n");
return;
}
}
//Contact 打印联系人的信息
void ShowContact( const Contact* pc)
{
assert(pc);//断言,防止pc是空指针
//判断通讯录中是否有联系人
if (pc->count == 0)
{
return;
}
else
{
printf("%-20s\t%-10s\t%-10s\t%-12s\t%-30s\n",
"姓名","年龄","性别","电话","地址");
int i = 0;
for(i = 0; i < pc->count; i++)
{
printf("%-20s\t%-10d\t%-10s\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);
}
printf("打印完成!\n");
return;
}
}
//查找,static 修饰的函数,只能在contact.c 文件中用,属于静态变量
static int FindByName(Contact* pc, const char* name)
{
assert(pc && name);
int i = 0;
for (i = 0; i < pc->count; i++)
{
if (0 == strcmp(pc->data[i].name, name))
{
return i;//找到
}
}
return -1;
}
//删除通讯录的某一个联系人
void DelContact( Contact* pc)
{
assert(pc);
char name[MAX_NAME] = { 0 };
int i = 0;
//判断通讯录里是否有联系人
if (pc->count == 0)
{
printf("通讯录为空!\n");
return;
}
else
{
//找到通讯录里你要删除的人
printf("请输入你要删除联系人的姓名:>");
scanf("%s", name);
//查找
//封装一个函数,去查找,找到,返回下标,找不到返回-1
int ret = FindByName(pc, name);
if (ret == -1)
{
printf("你要删除的联系人不存在!\n");
return;
}
else
{
//pc->count - 1 的原因是:加入count = 100,
//则 i = 99时,i + 1 == 100,会造成指针越界
for (i = ret; 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 };
int i = 0;
if (pc->count == 0)
{
printf("通讯录里没有联系人!\n");
return;
}
//查找
else
{
printf("请输入有查找的联系人:>");
scanf("%s", name);
//找到返回下标,不找到返回-1
int ret = FindByName(pc, name);
if (ret == -1)
{
printf("通讯录里不存在这个联系人!\n");
return;
}
else
{
//打印出来看一下
i = ret;
printf("%-20s\t%-10s\t%-10s\t%-12s\t%-30s\n",
"姓名", "年龄", "性别", "电话", "地址");
printf("%-20s\t%-10d\t%-10s\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);
}
printf("查找已完成!\n");
}
}
//修改指定联系人的信息
void ModifyContact(Contact* pc)
{
assert(pc);
char name[MAX_NAME] = { 0 };
if (pc->count == 0)
{
printf("通讯录是空的,没有联系人!\n");
}
else
{
printf("请输入你要修改信息的联系人:>");
scanf("%s", name);
//找到联系人
//不存在返回-1,存在返回下标
int ret = FindByName(pc, name);
if (ret == -1)
{
printf("你要修改的联系人不存在!\n");
return;
}
//修改
else
{
printf("要修改人已找到,接下来修改!\n");
printf("请输入姓名:>");
scanf("%s", pc->data[ret].name);//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");
}
int cmp_peo_by_name(const void* e1, const void* e2)
{
return(strcmp(((PeoInfo*)e1)->name,((PeoInfo*)e2) ->name));
}
//排序通讯录里面的联系人,按照名字来排
void SortContact(Contact* pc)
{
assert(pc);
if (pc->count == 0)
{
printf("通讯录是空的,没有联系人!\n");
return;
}
else
{
qsort(pc->data, pc->count, sizeof(PeoInfo), cmp_peo_by_name);
}
printf("排序成功!\n");
}
//清空所有联系人
void EmptyContact(Contact* pc)
{
//memset(pc->data, 0,sizeof(pc->data));
pc->count = 0;
printf("清空成功!\n");
return;
}