首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >基于静态顺序表实现通讯录

基于静态顺序表实现通讯录

作者头像
走在努力路上的自己
发布2024-01-26 17:22:12
发布2024-01-26 17:22:12
17200
代码可运行
举报
运行总次数:0
代码可运行

一、设计框架

test.c:通讯录的总体逻辑,测试通讯录的相关功能 contact.c:通讯录的实现模块 contact.h:通讯录的各种声明,包括库函数、自定义函数以及自定义结构体的声明

1、功能要求​

  1. 至少能够存储100个人的通讯信息​
  2. 能够保存用户信息:名字、性别、年龄、电话、地址等​
  3. 增加联系人信息​
  4. 删除指定联系人​
  5. 查找制定联系人​
  6. 修改指定联系人​
  7. 显示联系人信息​

2、菜单函数的实现

实现通讯录建立一个菜单是很重要的,并且菜单要包含通讯录所有的功能,以便于用户的操作

代码语言:javascript
代码运行次数:0
运行
复制
//通讯录菜单
void menu() {
	printf("*****************通讯录***************\n");
	printf("*******1.添加联系人  2.删除联系人*****\n");//ctrl+d
	printf("*******3.修改联系人  4.查找联系人*****\n");//ctrl+d
	printf("*******5.查看通讯录  0.  退 出  ******\n");//ctrl+d
	printf("**************************************\n");
}

二、头文件实现​

问题:头文件包含嵌套

解决方案:前置声明

Contact.h

代码语言:javascript
代码运行次数:0
运行
复制
#pragma once
//#include<stdio.h> //暂时加上

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

SeqList.h

代码语言:javascript
代码运行次数:0
运行
复制
#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#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);

三、Test.h

我们需要根据菜单里面的选项来选择进行我们需要实现的功能,比如我们想添加联系人,我们就输入1就会进行用户假如的操作,我们想退出程序我们输入0就可以退出。可以利用switch选择语句来实现各自的功能。

代码语言:javascript
代码运行次数:0
运行
复制
//#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;
}

四、通讯录的初始化和销毁

  • ContactInit:初始化通讯录。它调用SLInit来确保通讯录(被视为顺序表)的内部状态是干净的,即没有分配内存且大小与容量均为0。
  • ContactDesTroy:销毁通讯录。它调用SLDestroy来释放通讯录占用的内存资源,并确保通讯录回到初始状态。
  • SLInit:初始化顺序表。它将顺序表的数组指针设为NULL,大小与容量均设为0,表示这是一个空表。
  • SLDestroy:销毁顺序表。它首先检查顺序表指针是否为空,然后释放数组内存,将数组指针设为NULL,并将大小与容量重置为0。
代码语言:javascript
代码运行次数:0
运行
复制
//通讯录的初始化和销毁
//SL* ps
void ContactInit(Contact* pcon) {
	SLInit(pcon);
}

void ContactDesTroy(Contact* pcon) {
	SLDestroy(pcon);
}
代码语言:javascript
代码运行次数:0
运行
复制
//初始化和销毁
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函数):

  • 首先,定义了一个`Info`类型的结构体变量`info`,用于存储输入的联系人信息。
  • 然后,通过一系列`printf`和`scanf`函数,从用户那里获取联系人的姓名、年龄、性别、电话和住址,并保存到`info`结构体中。
  • 最后,调用`SLPushBack`函数,将这个联系人的信息添加到通讯录(实际上是一个顺序表)中。

2. 在顺序表尾部插入数据(SLPushBack函数):

  • 首先,使用`assert`函数检查传入的顺序表指针`ps`是否为空。如果为空,程序将终止运行。这是一种防止程序出错的方式。
  • 然后,调用`SLCheckCapacity`函数检查顺序表的容量是否足够。如果不够,该函数可能会进行扩容操作。
  • 最后,如果顺序表的空间足够,就直接在尾部插入数据,并更新顺序表的大小。
代码语言:javascript
代码运行次数:0
运行
复制
//增加通讯录
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);
}
代码语言:javascript
代码运行次数:0
运行
复制
//顺序表的头部/尾部插入
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++;
}

六、在通讯录中查找姓名下标

  • 使用 for 循环遍历通讯录中的每一个联系人。循环变量 i 从 0 开始,直到通讯录的大小 pcon->size。
  • 在每次循环中,使用 strcmp 函数比较当前联系人的姓名 pcon->arr[i].name 和要查找的姓名 name。strcmp 函数用于比较两个字符串是否相同。
  • 如果 strcmp 函数返回 0,表示找到了姓名匹配的联系人。此时,直接返回该联系人在通讯录中的下标 i。
  • 如果循环结束后仍然没有找到匹配的联系人,函数返回 -1,表示查找失败。
代码语言:javascript
代码运行次数:0
运行
复制
//在通讯录中查找姓名下标
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;
}

七、删除通讯录

  • 首先,函数会提示用户输入要删除的联系人的姓名。
  • 然后,它会调用 FindByName 函数(这个函数在之前的解释中已经介绍过,它的作用是在通讯录中查找指定姓名的联系人的下标)。
  • 如果 FindByName 函数返回的下标小于0,说明通讯录中没有这个姓名的联系人,函数就会打印一条提示信息,然后结束。
  • 如果找到了这个姓名的联系人(即 FindByName 返回的下标非负),函数就会调用 SLErase 函数来执行删除操作。
  • 删除操作成功后,函数会打印一条提示信息,告知用户联系人已经成功删除。
代码语言:javascript
代码运行次数:0
运行
复制
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");
}
  • 函数首先会检查传入的顺序表指针和位置参数是否有效。如果顺序表指针为空,或者位置参数超出了顺序表的大小范围,函数就会通过 assert 宏终止程序运行,防止发生错误。
  • 如果参数有效,函数就会从指定位置开始,将后面每个位置的数据都向前移动一个位置,覆盖掉当前位置的数据。这样就相当于删除了指定位置的数据。
  • 数据移动完成后,函数会将顺序表的大小减1,因为已经删除了一个数据。
代码语言:javascript
代码运行次数:0
运行
复制
//删除指定位置数据
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--;
}

八、显示通讯录

代码语言:javascript
代码运行次数:0
运行
复制
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
        );
    }
}

九、查找通讯录

  • 调用 FindByName 函数(该函数在之前的代码段中定义),在通讯录中查找指定姓名的联系人的下标。将返回的下标存储在 findIndex 变量中。
  • 判断 findIndex 的值:
  • 如果 findIndex 小于 0,表示没有找到匹配的联系人,于是打印一条提示信息并结束函数。
  • 如果 findIndex 非负,表示找到了匹配的联系人,继续执行下一步。
  • 打印找到的联系人的信息:
  • 首先打印一列标题,包括“姓名”、“性别”、“年龄”、“电话”和“住址”。
  • 然后根据 findIndex 下标,从通讯录中获取对应联系人的信息,并打印出来。
代码语言:javascript
代码运行次数:0
运行
复制
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
	);
}

今天就先到这了!!!

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-01-26,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、设计框架
    • 1、功能要求​
    • 2、菜单函数的实现
  • 二、头文件实现​
    • Contact.h
    • SeqList.h
  • 三、Test.h
  • 四、通讯录的初始化和销毁
  • 五、增加通讯录
  • 六、在通讯录中查找姓名下标
  • 七、删除通讯录
  • 八、显示通讯录
  • 九、查找通讯录
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档