【C语言】实现通讯录管理系统

这篇具有很好参考价值的文章主要介绍了【C语言】实现通讯录管理系统。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

大家好,我是苏貝,本篇博客带大家实现通讯录,如果你觉得我写的还不错的话,可以给我一个赞👍吗,感谢❤️
【C语言】实现通讯录管理系统,c语言,开发语言


一. 前言

本文将使用C语言来实现通讯录管理系统,该通讯录包括若干联系人,和每个联系人的姓名、年龄、性别、电话、地址。此通讯录的功能包括:增加联系人信息,删除联系人信息,查找联系人信息,修改联系人信息,显示所有联系人信息,根据名字顺序给所有联系人信息排序

二. 通讯录的实现

首先,我们建立1个contact.h文件,即一个头文件,用来声明。建立1个test.c文件,用来测试代码。再建立1个contact.c文件,用来写实现上述6个功能的代码

2.1 写出基本框架

下面的代码就是通讯录的基本框架,很多程序的基本框架和下面代码相似,用do-while循环来决定实现什么功能。变量input作为循环的判断条件,如果选择0,则退出循环,选择1~6之间的数则执行对应的功能,选择其它数则提示输入错误,再进入循环

int main()
{
	int input = 0;
	do
	{
		menu();
		printf("请选择:");
		scanf("%d", input);
		switch (input)
		{
		case 1:
			break;
		case 2:
			break;
		case 3:
			break;
		case 4:
			break;
		case 5:
			break;
		case 6:
			break;
		case 0:
			printf("退出程序\n");
			break;
		default:
			printf("选择错误,请重新输入\n");
			break;
		}
	} while (input);
	return 0;
}

2.2 制作menu菜单

基本框架中有menu菜单,那让我们来了解一下如何写menu菜单,即一个名为menu的函数。菜单界面就像餐厅里服务员递给你的菜单,根据菜单里的内容进行选择。选择1~6,实现特定功能;选择0,退出程序

void menu()
{
	printf("-----------------------------------\n");
	printf("***********************************\n");
	printf("******  1.Add       2.Del    ******\n");
	printf("******  3.Select    4.Modify ******\n");
	printf("******  5.Show      6.Sort   ******\n");
	printf("******  0.exit               ******\n");
	printf("***********************************\n");
	printf("-----------------------------------\n");
}

2.3 创建联系人和通讯录结构体

因为联系人信息包括姓名、年龄、性别、电话、地址,而且它们的类型和所占大小不同,所以我们想到构建一个联系人结构体,为了书写分别,我们将struct PeoInfo重命名为PeoInfo。

typedef struct PeoInfo
{
	char name[20];
	int age;
	char gender[5];
	char tele[12];
	char addr[30];
}PeoInfo;

现在我们将上面字符数组都确定了最大元素个数,那如果有一天我们发现有一个人的地址不止30个字节怎么办呢?我们接下来的代码里面也可能有addr[30],所以以后修改起来不太方便。因此,不妨我们用define定义一些标识符常量,之后敲代码时需要用到字符大小就写标识符常量而非一个数字,这样可以方便以后修改数组的大小

#define NAME_MAX 20
#define GENDER_MAX 5
#define TELE_MAX 30
#define ADDR_MAX 30

typedef struct PeoInfo
{
	char name[NAME_MAX];
	int age;
	char gender[GENDER_MAX];
	char tele[TELE_MAX];
	char addr[ADDR_MAX];
}PeoInfo;

如果我们想存储若干个联系人的信息,那我们就要定义一个数组,数组有若干个元素,元素的类型是PeoInfo,很自然的,我们用define定义的标识符常量来表示数组的最大元素个数。通讯录除了要有所有联系人信息外,也应该可以告诉我们有多少个联系人,所以通讯录里要有2个东西,一个是若干联系人信息,一个是联系人的个数,它们的类型不同,所以用结构体表示通讯录

#define MAX 1000
typedef struct Contact
{
	PeoInfo data[MAX];
	int sz;
}Contact;

2.3 的都是声明语句,所以都放在contact.h文件中


2.4 定义通讯录并初始化

通讯录的定义简简单单,不用多说

	Contact con;

我们来想想如何将它初始化。(如果不初始化,那么con里面的内容都是随机值)
在test.c文件中调用初始化函数,在contact.h中声明初始化函数,在contact.c文件中定义函数。结构体传参时最好是传址而非传值,且初始化需要改变结构体的内容,所以只能传址

	InitContact(&con);

初始化就是将变量con的元素sz的值赋值为0,并且data数组里面的元素也都是0。sz不用多说,将data数组里面的元素全变为0,只需应用我们之前介绍过的memset函数了解memset函数

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

2.5 使用枚举常量优化代码

如果不让你看菜单,你能知道上面基本框架的switch语句中的case 1~6分别是什么意思吗?如果你和我一样不太记得的话,那我们可以使用枚举常量来帮助我们识别。枚举常量如果我们自己不去主动赋值的话,它们的值会从0开始,依次+1,所以EXIT的值为0,ADD的值为1……

enum Option
{
	EXIT,//0
	ADD,
	DEL,
	SELECT,
	MODIFY,
	SHOW,
	SORT
};

所以switch语句中的case语句后面的值可以用带提示的枚举常量代替

switch (input)
		{
		case ADD:
			break;
		case DEL:
			break;
		case SELECT:
			break;
		case MODIFY:
			break;
		case SHOW:
			break;
		case SORT:
			break;
		case EXIT:
			printf("退出程序\n");
			break;
		default:
			printf("选择错误,请重新选择\n");
			break;
		}

2.6 增加联系人信息

在test.c文件中调用函数

case ADD:
			AddContact(&con);
			break;

在contact.c文件中定义函数。先判断通讯录是否满了,若没满则输入联系人信息,最好一定要记得sz++。注意:输入年龄的scanf语句中,记得要&

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

	pc->sz++;
	printf("增加成功\n");
}

2.7 删除联系人信息

在test.c文件中调用函数

case DEL:
			DelContact(&con);
			break;

在contact.c文件中定义函数。先判断通讯录是否为空,不为空时,输入要删除的人的姓名,在通过for循环搜索是否有该姓名的联系人,因为除了删除联系人信息功能需要搜索以外,查找和修改联系人信息也需要通过for循环搜索是否有该姓名的联系人,所以不妨将其封装成一个函数。若没有找到这个姓名的联系人,返回-1,否则返回该姓名的下标。
找到该姓名的联系人以后做删除操作,删除操作只需要让下标后面的元素依次向前进一位,覆盖掉下标的位置即可。所以循环的结束条件是i < pc->sz - 1,不要搞错。覆盖时只需要pc->data[i] = pc->data[i + 1];因为如果结构体的类型完全相同,那就可以整体赋值。最后sz–。可能有人注意到了,这个删除操作不能删除最后一个元素,是的,但没有关系,因为我们的sz–了,所以之后的遍历中不会再遍历原本的最后一个元素

int FindOfName(char* name, Contact* pc)
{
	assert(name && pc);
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		if (0 == strcmp(name, (pc->data[i].name)))
			return i;
	}
	return -1;
}

void DelContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX] = { 0 };
	if (pc->sz == 0)
	{
		printf("通讯录为空,删除失败\n");
		return;
	}
	printf("请输入要删除的人的名字:");
	scanf("%s", name);
	int ret = FindOfName(name, pc);
	if (ret == -1)
	{
		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");
}


2.8 查找联系人信息

查找联系人信息同样需要搜索函数FindOfName(name, pc);,如果要查找的人存在,那么会返回它的下标,再通过下标对这个人信息进行打印。为了直观的看出各个信息是属于哪一类的,我们可以在打印前打印提示信息,注意:年龄是%-5s,后面打印信息时是%-5d

void SelectContact(Contact* pc)
{
	char name[NAME_MAX] = { 0 };
	printf("请输入要查找的人的名字:");
	scanf("%s", name);
	int ret = FindOfName(name, pc);
	if (ret == -1)
	{
		printf("找不到要查找的人\n");
		return;
	}
	printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[ret].name, pc->data[ret].age,
		pc->data[ret].gender, pc->data[ret].tele, pc->data[ret].addr);
}

2.9 修改联系人信息

修改联系人信息同样需要搜索函数FindOfName(name, pc);,如果要想修改的人存在,那么会返回它的下标,再通过下标对这个人的全部信息重新输入

void ModifyContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX] = { 0 };
	printf("请输入要修改的人的名字:");
	scanf("%s", name);
	int ret = FindOfName(name, pc);
	if (ret == -1)
	{
		printf("找不到要修改的人\n");
		return;
	}
	printf("请输入名字:");
	scanf("%s", pc->data[ret].name);
	printf("请输入年龄:");
	scanf("%d", &(pc->data[ret].age));
	printf("请输入性别:");
	scanf("%s", pc->data[ret].gender);
	printf("请输入电话:");
	scanf("%s", pc->data[ret].tele);
	printf("请输入地址:");
	scanf("%s", pc->data[ret].addr);
}

2.10 打印所有联系人信息

打印所有联系人信息,无非就是遍历并打印所有元素,同样加上提示信息

void ShowContact(Contact* pc)
{
	assert(pc);
	if (pc->sz == 0)
	{
		printf("通讯录为空,打印无意义\n");
		return;
	}
	int i = 0;
	printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[i].name, pc->data[i].age,
			pc->data[i].gender, pc->data[i].tele, pc->data[i].addr);
	}
}

2.11 以名字排序所有联系人

使用qsort函数可快速排序,了解qsort函数

int cmp(const void* a, const void* b)
{
	return strcmp((char*)a, (char*)b);
}

void SortContact(Contact* pc)
{
	assert(pc);
	qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);
	printf("排序成功\n");
}

三. 模块化代码实现

3.1 test.c

#include"contact.h"

void menu()
{
	printf("-----------------------------------\n");
	printf("***********************************\n");
	printf("******  1.Add       2.Del    ******\n");
	printf("******  3.Select    4.Modify ******\n");
	printf("******  5.Show      6.Init   ******\n");
	printf("******  7.sort      0.exit   ******\n");
	printf("***********************************\n");
	printf("-----------------------------------\n");
}

enum Option
{
	EXIT,
	ADD,
	DEL,
	SELECT,
	MODIFY,
	SHOW,
	INIT,
	SORT
};

int main()
{
	Contact con;
	//初始化通讯录
	InitContact(&con);
	int input = 0;
	do
	{ 
		menu();
		printf("请选择:");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContact(&con);
			break;
		case DEL:
			DelContact(&con);
			break;
		case SELECT:
			SelectContact(&con);
			break;
		case MODIFY:
			ModifyContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case INIT:
			InitContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case EXIT:
			printf("退出程序\n");
			break;
		default:
			printf("选择错误,请重新选择\n");
			break;
		}
	} while (input);
	return 0;
}

3.2 contact.h

#pragma once

#include<stdio.h>
#include<assert.h>
#include<string.h>


#define NAME_MAX 20
#define GENDER_MAX 5
#define TELE_MAX 30
#define ADDR_MAX 30
#define MAX 1000


typedef struct PeoInfo
{
	char name[NAME_MAX];
	int age;
	char gender[GENDER_MAX];
	char tele[TELE_MAX];
	char addr[ADDR_MAX];
}PeoInfo;



typedef struct Contact
{
	PeoInfo data[MAX];
	int sz;
}Contact;


//初始化通讯录
void InitContact(Contact* pc);

//增加联系人
void AddContact(Contact* pc);

//删除联系人
void DelContact(Contact* pc);

//打印通讯录
void ShowContact(Contact* pc);

//查找联系人
void SelectContact(Contact* pc);

//修改联系人信息
void ModifyContact(Contact* pc);

//以名字排序所有联系人
void SortContact(Contact* pc);

3.3 contact.c

#include"contact.h"

//初始化通讯录
void InitContact(Contact* pc)
{
	assert(pc);
	pc->sz = 0;
	memset(pc->data, 0, sizeof(pc->data));
}

//增加联系人
void AddContact(Contact* pc)
{
	assert(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].gender);
	printf("请输入电话:");
	scanf("%s", pc->data[pc->sz].tele);
	printf("请输入地址:");
	scanf("%s", pc->data[pc->sz].addr);

	pc->sz++;
	printf("增加成功\n");
}


int FindOfName(char* name, Contact* pc)
{
	assert(name && pc);
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		if (0 == strcmp(name, (pc->data[i].name)))
			return i;
	}
	return -1;
}

//删除联系人
void DelContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX] = { 0 };
	if (pc->sz == 0)
	{
		printf("通讯录为空,删除失败\n");
		return;
	}
	printf("请输入要删除的人的名字:");
	scanf("%s", name);
	int ret = FindOfName(name, pc);
	if (ret == -1)
	{
		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 ShowContact(Contact* pc)
{
	assert(pc);
	if (pc->sz == 0)
	{
		printf("通讯录为空,打印无意义\n");
		return;
	}
	int i = 0;
	printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[i].name, pc->data[i].age,
			pc->data[i].gender, pc->data[i].tele, pc->data[i].addr);
	}
}



//查找联系人
void SelectContact(Contact* pc)
{
	char name[NAME_MAX] = { 0 };
	printf("请输入要查找的人的名字:");
	scanf("%s", name);
	int ret = FindOfName(name, pc);
	if (ret == -1)
	{
		printf("找不到要查找的人\n");
		return;
	}
	printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[ret].name, pc->data[ret].age,
		pc->data[ret].gender, pc->data[ret].tele, pc->data[ret].addr);
}



//修改联系人信息
void ModifyContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX] = { 0 };
	printf("请输入要修改的人的名字:");
	scanf("%s", name);
	int ret = FindOfName(name, pc);
	if (ret == -1)
	{
		printf("找不到要修改的人\n");
		return;
	}
	printf("请输入名字:");
	scanf("%s", pc->data[ret].name);
	printf("请输入年龄:");
	scanf("%d", &(pc->data[ret].age));
	printf("请输入性别:");
	scanf("%s", pc->data[ret].gender);
	printf("请输入电话:");
	scanf("%s", pc->data[ret].tele);
	printf("请输入地址:");
	scanf("%s", pc->data[ret].addr);
}


int cmp(const void* a, const void* b)
{
	return strcmp((char*)a, (char*)b);
}


//以名字排序所有联系人
void SortContact(Contact* pc)
{
	assert(pc);
	qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);
	printf("排序成功\n");
}

3.4 结果演示

【C语言】实现通讯录管理系统,c语言,开发语言
【C语言】实现通讯录管理系统,c语言,开发语言
【C语言】实现通讯录管理系统,c语言,开发语言
【C语言】实现通讯录管理系统,c语言,开发语言
【C语言】实现通讯录管理系统,c语言,开发语言


四. 存在的问题

前面实现的通讯录有什么问题呢?
1.录入的信息,等程序结束后,还存在吗?不存在了,因为数据是存放内存中的,只要程序退出,掉电,都会
丢失怎么解决呢?用文件存储,数据库…
2.通讯录的大小是固定的100个元素,存放100个人的信息如果信息太多,空间小了;如果信息太少,空间大了。怎么解决?用动态内存管理!
所以期待博主的下一篇优化的通讯录管理系统吧


好了,那么本篇博客就到此结束了,如果你觉得本篇博客对你有些帮助,可以给个大大的赞👍吗,感谢看到这里,我们下篇博客见❤️文章来源地址https://www.toymoban.com/news/detail-717306.html

到了这里,关于【C语言】实现通讯录管理系统的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 文件操作介绍及C语言实现通讯录管理系统3.0最终版(文件操作版本)

    文件操作介绍及C语言实现通讯录管理系统3.0最终版(文件操作版本)

    上一篇文章我们学习了动态内存开辟的相关知识点,并用动态内存函数优化了我们的通讯录,但通讯录还有需要改进的地方,比如,正常情况下的通讯录,应该可以一直保存联系人信息,而不是退出就清空了,这就需要我们实实在在的保存下来一个通讯录。 接下来我会给大家

    2023年04月08日
    浏览(12)
  • 【C语言】优化通讯录管理系统

    【C语言】优化通讯录管理系统

    大家好,我是苏貝,本篇博客带大家优化上一篇的通讯录,如果你觉得我写的还不错的话,可以给我一个赞👍吗,感谢❤️ 在上一篇博文 实现通讯录管理系统 的最后,我们了解了那段代码的问题:通讯录的大小是固定的100个元素,如果想要存放1000个人的信息,那么空间就

    2024年02月06日
    浏览(42)
  • c语言:通讯录管理系统(文件版本)

    c语言:通讯录管理系统(文件版本)

    前言: 本通讯录管理系统一共三个版本,除此文章以外还有如下俩个版本,大家可以根据需求自取: 基础增删查改功能版本 :c语言:通讯录管理系统(增删查改)-CSDN博客 动态分配内存版本 :c语言:通讯录管理系统(动态分配内存版)-CSDN博客         本文是在基础的通

    2024年02月07日
    浏览(39)
  • 【C语言】优化通讯录管理系统2

    【C语言】优化通讯录管理系统2

    本篇博客是基于上一篇博客写出来的,了解上一篇博客 大家好,我是苏貝,本篇博客带大家再次优化上一篇的通讯录,实现将录入的数据在程序退出后存储到文件中,在下一次程序开始时打开文件获取数据,如果你觉得我写的还不错的话,可以给我一个赞👍吗,感谢❤️ 既

    2024年02月04日
    浏览(40)
  • c语言:通讯录管理系统(增删查改)

    c语言:通讯录管理系统(增删查改)

      前言: 本通讯录管理系统一共三个版本,除此文章以外还有如下俩个版本,大家可以根据需求自取: 动态分配内测版 :c语言:通讯录管理系统(动态分配内存版)-CSDN博客 文件版本 :c语言:通讯录管理系统(文件版本)-CSDN博客 目录 一.大体的框架 主函数初步设计 联系

    2024年02月08日
    浏览(42)
  • C语言课程设计_通讯录管理系统

    C语言课程设计_通讯录管理系统

    以本班同学的具体数据为背景,设计一个本班同学通讯录。 1)  通讯录编辑(添加、删除)。 2)  按不同的项进行查找。 3)  对已存在的通讯录按不同的项排序。 4)  将通讯录写入文件 5)从文件读入通讯录 备注:通讯录至少应该有以下数据项: 姓名,地址,电话,邮编,

    2024年02月06日
    浏览(40)
  • 【C语言】通讯录管理系统(附图解、源码)

    【C语言】通讯录管理系统(附图解、源码)

    目录 一.前言 二.准备工作 三.ContactTest.c测试区 1.菜单 2.选择功能 四.Contact.h头文件引用区 1.通讯录成员结构体函数的创建 2.实现功能函数的创建 五.ContactRealize.c功能实现区 1.初始化成员信息 2.查找目标成员位置 3.增加联系人 4.删除指定联系人 5.查找指定联系人 6.修改指定联系

    2024年02月03日
    浏览(42)
  • c语言:通讯录管理系统(动态分配内存版)

    c语言:通讯录管理系统(动态分配内存版)

    前言: 本通讯录管理系统一共三个版本,除此文章以外还有如下俩个版本,大家可以根据需求自取: 基础增删查改功能版本 :c语言:通讯录管理系统(增删查改)_luming.02的博客-CSDN博客 文件保存版本 :c语言:通讯录管理系统(文件版本)-CSDN博客         本文是在基

    2024年02月08日
    浏览(50)
  • C语言课程设计|通讯录管理系统(含完整代码)

    目录 菜单功能 录入联系人信息功能 查看系统中全部信息功能 查看单个信息功能 删除全部信息功能 删除单个信息功能 修改信息功能 完整代码 在长达一个多月的学习过程中,终于将C语言学完,因此专门写一个C语言课程设计来检验这一个多月的学习成果,由于写的比较急,

    2024年02月01日
    浏览(40)
  • 精通C语言:打造高效便捷的通讯录管理系统

    精通C语言:打造高效便捷的通讯录管理系统

    ✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:C语言项目 贝蒂的主页:Betty‘s blog 在我们大致学习完C语言之后,我们就可以利用目前所学的知识去做一些有意思的项目,而今天贝蒂就带大家完成一个通讯录的简易实现, 本章你可能需要

    2024年02月19日
    浏览(39)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包