目录
一、设计框架
1、功能要求
2、菜单函数的实现
二、头文件实现
Contact.h
SeqList.h
三、Test.h
四、通讯录的初始化和销毁
五、增加通讯录
六、在通讯录中查找姓名下标
七、删除通讯录
八、显示通讯录
九、查找通讯录
test.c:通讯录的总体逻辑,测试通讯录的相关功能
contact.c:通讯录的实现模块
contact.h:通讯录的各种声明,包括库函数、自定义函数以及自定义结构体的声明
实现通讯录建立一个菜单是很重要的,并且菜单要包含通讯录所有的功能,以便于用户的操作
//通讯录菜单
void menu() {
printf("*****************通讯录***************\n");
printf("*******1.添加联系人 2.删除联系人*****\n");//ctrl+d
printf("*******3.修改联系人 4.查找联系人*****\n");//ctrl+d
printf("*******5.查看通讯录 0. 退 出 ******\n");//ctrl+d
printf("**************************************\n");
}
问题:头文件包含嵌套
解决方案:前置声明
#pragma once
//#include //暂时加上
#define NAME_MAX 100
#define GENDER_MAX 10
#define TEL_MAX 12
#define ADDR_MAX 100
//通讯录数据类型
typedef struct PersonInfo
{
char name[NAME_MAX];
int age;
char gender[GENDER_MAX];
char tel[TEL_MAX];
char addr[ADDR_MAX];
}Info;
//使用顺序表的前置声明
struct SeqList;
typedef struct SeqList Contact;
//通讯里提供的操作
//通讯录的初始化和销毁
void ContactInit(Contact* pcon);//实际初始化的还是顺序表
void ContactDesTroy(Contact* pcon);
//增加、删除、修改、查找、查看通讯录
void ContactAdd(Contact* pcon);
void ContactDel(Contact* pcon);
void ContactModify(Contact* pcon);
void ContactFind(Contact* pcon);
void ContactShow(Contact* pcon);
#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include
#include
#include
#include"Contact.h"
//静态顺序表
//#define N 100
//struct SeqList
//{
// SLDataType a[N];
// int size;
//};
//动态顺序表
//typedef int SLDataType;
typedef Info SLDataType;
typedef struct SeqList
{
SLDataType* arr; //存储数据的底层结构
int capacity; //记录顺序表的空间大小
int size; //记录顺序表当前有效的数据个数
}SL;
//typedef struct SeqList SL;
//初始化和销毁
void SLInit(SL* ps);
void SLDestroy(SL* ps);
void SLPrint(SL* ps); //保持接口一致性
//顺序表的头部/尾部插入
void SLPushBack(SL* ps, SLDataType x);
void SLPushFront(SL* ps, SLDataType x);
//顺序表的头部/尾部删除
void SLPopBack(SL* ps);
void SLPopFront(SL* ps);
//指定位置之前插入数据
//删除指定位置数据
void SLInsert(SL* ps, int pos, SLDataType x);
void SLErase(SL* ps, int pos);
//int SLFind(SL* ps, SLDataType x);
我们需要根据菜单里面的选项来选择进行我们需要实现的功能,比如我们想添加联系人,我们就输入1就会进行用户假如的操作,我们想退出程序我们输入0就可以退出。可以利用switch选择语句来实现各自的功能。
//#include"Contact.h" //在SeqList.h文件中已经包了Contact.h
#include"SeqList.h"
//通讯录菜单
void menu() {
printf("*****************通讯录***************\n");
printf("*******1.添加联系人 2.删除联系人*****\n");//ctrl+d
printf("*******3.修改联系人 4.查找联系人*****\n");//ctrl+d
printf("*******5.查看通讯录 0. 退 出 ******\n");//ctrl+d
printf("**************************************\n");
}
int main()
{
int op = -1;
//创建通讯录结构对象
Contact con;
ContactInit(&con);
do {
menu();
printf("请选择您的操作:\n");
scanf("%d", &op);
switch (op)
{
case 1:
//添加联系人
ContactAdd(&con);
break;
case 2:
//删除联系人
ContactDel(&con);
break;
case 3:
//修改联系人
ContactModify(&con);
break;
case 4:
//查找联系人
ContactFind(&con);
break;
case 5:
//查看通讯录
ContactShow(&con);
break;
case 0:
//退出通讯录
printf("通讯录退出中...\n");
break;
default:
break;
}
} while (op != 0);
//销毁通讯录
ContactDesTroy(&con);
return 0;
}
//通讯录的初始化和销毁
//SL* ps
void ContactInit(Contact* pcon) {
SLInit(pcon);
}
void ContactDesTroy(Contact* pcon) {
SLDestroy(pcon);
}
//初始化和销毁
void SLInit(SL* ps) {
ps->arr = NULL; //不是int 而是Info类型
ps->size = ps->capacity = 0;
}
void SLDestroy(SL* ps) {
assert(ps);
if (ps->arr) {
free(ps->arr);
}
ps->arr = NULL;
ps->size = ps->capacity = 0;
}
这段代码主要实现了两个功能:添加联系人到通讯录(ContactAdd函数)和在顺序表尾部插入数据(SLPushBack函数)。
1. 添加联系人到通讯录(ContactAdd函数):
2. 在顺序表尾部插入数据(SLPushBack函数):
//增加通讯录
void ContactAdd(Contact* pcon) {
//创建联系人结构体变量
Info info;
printf("请输入联系人姓名:\n");
scanf("%s", info.name);
printf("请输入联系人年龄:\n");
scanf("%d", &info.age);
//其他数据都是数组,年龄不是
printf("请输入联系人性别:\n");
scanf("%s", info.gender);
printf("请输入联系人电话:\n");
scanf("%s", info.tel);
printf("请输入联系人住址:\n");
scanf("%s", info.addr);
//保存数据到通讯录(顺序表)
SLPushBack(pcon, info);
}
//顺序表的头部/尾部插入
void SLPushBack(SL* ps, SLDataType x) {
//断言--粗暴的解决方式
//assert(ps != NULL);
assert(ps);
//if判断--温柔的解决方式
//if (ps == NULL) {
// return;
//}
//空间不够,扩容
SLCheckCapacity(ps);
//空间足够,直接插入
ps->arr[ps->size++] = x;
//ps->size++;
}
//在通讯录中查找姓名下标
int FindByName(Contact* pcon, char name[]) {
for (int i = 0; i < pcon->size; i++)
{
if (strcmp(pcon->arr[i].name, name) == 0) {
//找到了,返回下标
return i;
}
}
return -1;
}
void ContactDel(Contact* pcon) {
//删除之前一定要先查找
//找到了,可以删除
//找不到,不能执行删除
printf("请输入要删除的联系人姓名:\n");
char name[NAME_MAX];
scanf("%s", name);
int findIndex = FindByName(pcon, name);
if (findIndex < 0) {
printf("要删除的联系人不存在!\n");
return;
}
//执行删除操作
SLErase(pcon, findIndex);
printf("联系人删除成功!\n");
}
//删除指定位置数据
void SLErase(SL* ps, int pos) {
assert(ps);
assert(pos >= 0 && pos < ps->size);
//pos以后的数据往前挪动一位
for (int i = pos; i < ps->size - 1; i++)
{
ps->arr[i] = ps->arr[i + 1];//ps->arr[i-2] = ps->arr[i-1];
}
ps->size--;
}
void ContactShow(Contact* pcon) {
//格式大家下去感兴趣去调整
printf("%s %s %s %s %s\n", "姓名", "性别", "年龄", "电话", "住址");
for (int i = 0; i < pcon->size; i++)
{
printf("%s %s %d %s %s\n",
pcon->arr[i].name,
pcon->arr[i].gender,
pcon->arr[i].age,
pcon->arr[i].tel,
pcon->arr[i].addr
);
}
}
void ContactFind(Contact* pcon) {
char name[NAME_MAX];
printf("请输入要查找的用户姓名:\n");
scanf("%s", name);
int findIndex = FindByName(pcon, name);
if (findIndex < 0) {
printf("该联系人不存在!\n");
return;
}
//找到了,打印一下查找的联系人信息
printf("%s %s %s %s %s\n", "姓名", "性别", "年龄", "电话", "住址");
printf("%s %s %d %s %s\n",
pcon->arr[findIndex].name,
pcon->arr[findIndex].gender,
pcon->arr[findIndex].age,
pcon->arr[findIndex].tel,
pcon->arr[findIndex].addr
);
}
今天就先到这了!!!
看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!
你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。
我的博客即将同步至腾讯云开发者社区,邀请大家一同入驻:
腾讯云自媒体分享计划 - 腾讯云开发者社区-腾讯云