【C语言】通讯录实现

2023-05-16

通讯录功能

  1. 添加联系人信息(名字,性别,年龄,电话号码,家庭住址)
  2. 输出指定联系人信息
  3. 查找指定联系人信息
  4. 修改指定联系人信息
  5. 打印所有联系人信息
  6. 对所有联系人(通过名字)排序
  7. 保存当前通讯录内容(以二进制形式保存到当前文件夹中的contactinformation这个数据文件里)

文件分装

在这里插入图片描述

对应文件的代码

contact.h

//声明C库函数的头文件
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<errno.h>

//定义相关量的宏
//struct PeoInfo数组成员的默认大小 
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30

//初始化时,通讯录的默认容量大小
#define DEFAULT_SIZE 3

//枚举,作为case语句的执行常量表达式
enum Option
{
	EXIT,//0
	ADD,//1
	DEL,//2
	SEARCH,//3
	MODIFY,//4
	SHOW,//5
	SORT,//6
	SAVE//7
};

//声明成员信息的结构体
struct PeoInfo
{
	char name[MAX_NAME];
	char sex[MAX_SEX];
	int age;
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
};

//声明通讯录信息的结构体
struct Contact
{
	struct PeoInfo* data;//存放所有成员的数据
	int size;//记录通讯录当前有多少人
	int capacity;//记录通讯录当前容量大小
};

//函数声明
void InitContact(struct Contact* p);
void AddContact(struct Contact* p);
void ShowContact(struct Contact* p);
void SearchContact(struct Contact* p);
void DelContact(struct Contact* p);
void ModifyContact(struct Contact* p);
void SortContact(struct Contact* p);
void DistroyContact(struct Contact* p);
void SaveContact(struct Contact* p);

contact.c

#include"contact.h"

//检查通讯录的容量(内存空间)是否足够
//不够就在开辟两个成员空间
//够了就什么都不做
static void CheckCapacity(struct Contact* p)
{
	//判断当前通讯录容量是否满了
	if (p->size == p->capacity)//当前通讯录成员人数是否等于最大容量
	{
		//开辟空间
		struct PeoInfo* tmp = (struct PeoInfo*)realloc(p->data, (2 + p->capacity) * sizeof(struct PeoInfo));
		if (tmp != NULL)//开辟成功
		{
			p->data = tmp;
			p->capacity += 2;
		}
		else//开辟失败
		{
			printf("CheckCapacity::%s\n", strerror(errno));
			return;
		}
	}
}

//上传上次写在文件里的成员信息
static void LoadContact(struct Contact* p)
{
	//以只读的方式打开文件
	FILE* pf = fopen("contactinformation", "rb");
	if (pf == NULL)//检查是否打开
	{
		printf("LoadContact::%s\n", strerror(errno));
	}
	struct PeoInfo tmp = { 0 };//创建一个临时的结构体变量
	//开始逐条读取pf的(成员信息)内容放到中间变量tmp里
	while (fread(&tmp, sizeof(struct PeoInfo), 1, pf))
	{
		CheckCapacity(p);//检查当前的结构体容量是否可以放得下所有之前的成员信息
		*(p->data + p->size) = tmp;
		p->size++;//每放一个成员,记录成员人数的size就加一
	}
	//文件读取完毕,关闭文件
	fclose(pf);
	pf = NULL;
}

//初始化通讯录信息
void InitContact(struct Contact* p)
{
	//为通讯录DEFAULT_SIZE个成员的空间
	p->data = (struct PeoInfo*)malloc(DEFAULT_SIZE * sizeof(struct PeoInfo));
	if (p->data == NULL)//检查是否开辟成功
	{
		printf("InitContact::%s\n", strerror(errno));
		return;
	}
	p->size = 0;
	p->capacity= DEFAULT_SIZE;
	LoadContact(p);//初始化之后把以前写在文件里的成员信息上次到这次程序的通讯录里
}


//增加成员信息
void AddContact(struct Contact* p)
{
	void CheckCapacity(p);//检查当前通讯录容量是否足够,不够的话就开辟空间,够了就什么都不干

	printf("请输入名字:>");
	scanf("%s",(*((p->data)+(p->size))).name);

	printf("请输入性别:>");
	scanf("%s", (*((p->data) + (p->size))).sex);

	printf("请输入年龄:>");
	scanf("%d", &(*((p->data) + (p->size))).age);

	printf("请输入电话号码:>");
	scanf("%s", (*((p->data) + (p->size))).tele);

	printf("请输入住址:>");
	scanf("%s", (*((p->data) + (p->size))).addr);

	printf("添加成功\n");

	p->size++;//添加完成之后记录当前通讯录成员个数的size变量加一
}


//查看通讯录中所有成员信息
void ShowContact(struct Contact* p)
{
	//先检查当前通讯录是否有成员信息
	if (p->size == 0)//没有成员
	{
		printf("通讯录内容为空\n");
	}
	else//有成员,依次打印每个成员的信息
	{
		printf("%-20s\t%-4s\t%-5s\t%-12s\t%-20s\n", "名字", "性别", "年龄", "电话", "住址");
		int i = 0;
		for (i = 0; i < p->size; i++)
		{
			printf("%-20s\t%-4s\t%-5d\t%-12s\t%-20s\n",
				(*((p->data) + i)).name,
				(*((p->data) + i)).sex,
				(*((p->data) + i)).age,
				(*((p->data) + i)).tele,
				(*((p->data) + i)).addr);
		}
	}
}

//实现查找成员名字功能的函数
//找到了就返回成员的下标
//找不到就返回-1
static int Search_By_Name(struct Contact* p,char* pc)
{
	int i = 0;
	for (i = 0; i < p->size; i++)
	{
		if (strcmp((*((p->data) + i)).name, pc) == 0)
		{
			return i;
		}
	}
	return -1;
}

//查找指定成员
void SearchContact(struct Contact* p)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要查找人的姓名:>");
	scanf("%s", name);
	//判断该成员是否存在
	int flag = Search_By_Name(p, name);
	if (flag == -1)//不存在
	{
		printf("要查找人的信息不存在\n");
	}
	else//存在,打印该成员的信息
	{
		printf("%-20s\t%-4s\t%-5s\t%-12s\t%-20s\n", "名字", "性别", "年龄", "电话", "住址");

		printf("%-20s\t%-4s\t%-5d\t%-12s\t%-20s\n",
			(*((p->data) + flag)).name,
			(*((p->data) + flag)).sex,
			(*((p->data) + flag)).age,
			(*((p->data) + flag)).tele,
			(*((p->data) + flag)).addr);
	}
}

//输出通讯录中指定成员信息
void DelContact(struct Contact* p)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要删除人的名字:>");
	scanf("%s", name);
	//判断该成员是否存在
	int flag = Search_By_Name(p, name);
	if (flag == -1)//不存在
	{
		printf("要删除的人不存在\n");
	}
	else//存在,把后面成员的位置往前挪一位
	{
		int i = 0;
		for (i = flag; i < p->size-1; i++)
		{
			*(p->data + i) = *(p->data + i + 1);
		}
		printf("删除成功\n");
		p->size--;//令记录当前通讯录成员个数的size减一
	}
}

//修改通讯录中指定成员信息
void ModifyContact(struct Contact* p)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", name);
	//判断该成员是否存在
	int flag=Search_By_Name(p, name);
	if (flag == -1)//不存在
	{
		printf("要修改人的信息不存在\n");
	}
	else//存在,通过访问结构体成员直接修改信息
	{
		printf("请输入名字:>");
		scanf("%s", (*((p->data) + flag)).name);

		printf("请输入性别:>");
		scanf("%s", (*((p->data) + flag)).sex);

		printf("请输入年龄:>");
		scanf("%d", &(*((p->data) + flag)).age);

		printf("请输入电话号码:>");
		scanf("%s", (*((p->data) + flag)).tele);

		printf("请输入住址:>");
		scanf("%s", (*((p->data) + flag)).addr);

		printf("修改成功\n");
	}
}

//分隔行内都是实现(通过名字)排序功能的相关函数

//-------------------------------------------------------------------------------------------------------------------------------------------------------------

//自定义的比较函数(通过名字来比较两个元素的大小)
//比较两个元素
//p1>p2返回大于零的数字
//p1=p2返回0
//p1<p2返回小于零的数字
static int Cmp_Struct(void* p1, void* p2)
{
	return strcmp(((struct PeoInfo*)p1)->name , ((struct PeoInfo*)p2)->name);
}

//交换两个元素的内容(width,是一个元素所占内存空间的大小,单位是字节)
static void Swap(void* p1, void* p2, int width)
{
	//一个字节一个字节的交换
	while (width)
	{
		char tmp = *(char*)p1;
		*(char*)p1 = *(char*)p2;
		*(char*)p2 = tmp;
		++(char*)p1;
		++(char*)p2;
		width--;
	}
}

//通用的冒泡排序(什么类型的数据都可以排)
static void My_Bubble_Sort(void* p, int sz, int width, int(*cmp)(void*, void*))
{
	int i = 0;
	int j = 0;
	int flag = 1;//flag为1是是已经排好序,为0是是还没排好序,先默认它为1
	//冒泡排序
	for (i = 0; i < sz - 1; i++)//sz个元素,一共要比较sz-1趟,每一趟排序好一个元素
	{
		for (j = 0; j < sz - 1 - i; j++)//每趟要两两比较当前未排好序的元素个数-1次
		{
			if (cmp((char*)p + (j*width), (char*)p + ((j + 1)*width)) > 0)//判断是否要交换两个元素(既判断前一个元素的名字是否大于后一个元素的名字)
			{
				Swap((char*)p + (j*width), (char*)p + ((j + 1)*width), width);//交换两个成员的位置
				flag = 0;//已经判断这次是无序了,所以flag=0
			}
		}
		if (flag == 1)//若一趟两两比较下来flag仍然=1,则数组肯定有序了,跳出冒泡排序
		{
			break;
		}
	}
}

void SortContact(struct Contact* p)
{
	//排序之前的传参
	My_Bubble_Sort(p->data,p->size,sizeof(p->data[0]),Cmp_Struct);
	//p->data:成员信息数组的首元素地址
	//p->size:当前成员个数
	//sizeof(p->data[0]):一个成员所占内存空间的大小,单位是字节
	//Cmp_Struct:自定义的比较两个元素功能的函数
	printf("排序完成\n");
}

//-------------------------------------------------------------------------------------------------------------------------------------------------------------

//释放之前malloc开辟的空间
void DistroyContact(struct Contact* p)
{
	free(p->data);
	p->data = NULL;
	printf("退出通讯录\n");
}

//保存当前通讯录的成员信息到contactinformation这个文件里
void SaveContact(struct Contact* p)
{
	//以只写的方式打开文件
	FILE* pf = fopen("contactinformation", "wb");
	//判断文件是否打开
	if (pf == NULL)//打开失败的情况
	{
		printf("SaveContact::%s\n", strerror(errno));
		return;
	}
	//开始逐条读取成员信息放到contactinformation文件里
	int i = 0;
	for (i = 0; i < p->size; i++)
	{
		fwrite(p->data + i, sizeof(struct PeoInfo), 1, pf);
	}
	printf("保存成功\n");
	//关闭文件
	fclose(pf);
	pf = NULL;
}

test.c

#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.save              0.exit   *****\n");
	printf("****************************************\n");
}

//主函数
int main()
{
	struct Contact con = {0};
	InitContact(&con);
	int input = 0;
	do
	{
		menu();//显示通讯录功能
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case EXIT://0
			SaveContact(&con);
			DistroyContact(&con);
			break;
		case ADD://1
			AddContact(&con);
			break;
		case DEL://2
			DelContact(&con);
			break;
		case SEARCH://3
			SearchContact(&con);
			break;
		case MODIFY://4
			ModifyContact(&con);
			break;
		case SHOW://5
			ShowContact(&con);
			break;
		case SORT://6
			SortContact(&con);
			ShowContact(&con);
			break;
		case SAVE://7
			SaveContact(&con);
			break;
		default:
			printf("选择错误,请重新选择\n");
			break;
		}
	} while (input);
	return 0;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【C语言】通讯录实现 的相关文章

  • 运行mmsegmentation碰到报错:size shape must match input shape. Input is 2D, size is 3

    运行mmsegmentation碰到报错 xff1a size shape must match input shape Input is 2D size is 3 这个问题比较坑 有时候我们的标签size是 512 xff0c 512 x
  • (docker笔记):镜像原理之联合文件系统、分层理解、commit 镜像

    目录 Docker 镜像讲解 什么是镜像 xff1f Docker 镜像加载原理 分层理解 commit 镜像 Docker 镜像讲解 什么是镜像 xff1f 镜像是一种轻量级 可执行的独立软件包 xff0c 用来打包软件运行环境和基于运行
  • FCOS报错ImportError: libcudart.so

    在使用detectron2 43 AdelaiDet运行FCOS代码时 xff0c 碰到报错 xff1a from adet import C ImportError libcudart so 10 0 cannot open shared
  • 在Ubuntu18.04(Melodic)中安装ROS Gazebo经验总结

    首先实现Win10系统 与 Ubuntu18 04 之间实现粘贴复制 span class token function sudo span span class token function apt span span class tok
  • 解决ResourceNotFound: gazebo_ros

    roslaunch simulation landing px4 launch 错误如下 ResourceNotFound gazebo ros ROS path 0 61 opt ros kinetic share ros ROS pat
  • Ubuntu18.04下java环境安装flightplot分析软件

    Java安装参考我的这篇博文 Linux 系统小白 Ubuntu 18 04下安装 Java 过程简单总结 安装flightplot过程如下 1 xff1a sudo apt install git 2 xff1a git clone re
  • estimatePoseSingleMarkers()解读

    ex 网页长截图 1 F12 2 ctrl 43 shift 43 p 3 Capture full size screenshot 参考网址 estimatePoseSingleMarkers void cv aruco estimate
  • 插值函数总结(下篇之二维插值)

    命令2 interp2 功能 xff1a 二维数据内插值 xff08 表格查找 xff09 格式1 xff1a z1 61 interp2 x y z x1 y1 返回矩阵z1 xff0c 其元素包含对应于参量x1与y1 xff08 可以是
  • 模型预测控制算法(MPC算法)底层逻辑

    目录 MPC算法的基本原理 详细解析 预测模型中需要注意的点 滚动优化需要注意的点 构造目标函数约束部分 约束部分 举例说明 复盘总结 MPC算法的基本原理 MPC 的基本原理可以分为三个步骤 xff1a 预测模型 滚动优化 反馈校正 xf
  • 计算机视觉(三)--- 图像到图像的映射(全景拼接)

    目录 1 基本介绍 2 RANSAC 3 单应性矩阵估计 4 全景图像 1 基本介绍 引言 众所周知 xff0c 在我们拍摄风光摄影时 xff0c 广角镜头是使用频率最高的镜头 xff0c 特别是拍摄那些波澜壮阔的大场景风光 而且镜头可谓是
  • 机器学习课后练习题(期末复习题目附答案)

    此为第一章绪论部分 一 单选题 1 移动运营商对客户的流失进行预测 可以使用下面哪种机器学习方法比较合适 A 一元线性回归分析 B 关联方法 C 聚类算法 D 多层前馈网络 正确答案 A 2 下面哪种说法有关机器学习的认识是错误的 A 高质
  • STM32野火教程学习笔记

    欢迎使用STM32 虽然经历了疫情期间的价格起飞 xff0c 但是STM32系列的单片机仍然是各个控制领域内主流的微控制器 它是控制人的必修课之一 STM32的编程方法 我们在学习51单片机的时候 xff0c 通常是通过编写程序直接对其输入
  • Docker的使用

    目录 Docker概念 Docker安装 配置docker加速器 Docker命令 docker进程命令 docker镜像命令 容器命令 Docker容器的数据卷 数据卷概念 数据卷配置 数据卷容器 docker部署mysql 1 搜索my
  • launch启动文件的使用方法

    launch文件 xff1a 脚本 xff0c 可以把很多节点的启动写进去 Launch文件语法 1 lt launch gt launch文件中的根元素采用 lt launch gt 标签定义 2 lt node gt pkg xff1a
  • 基于单片机的电子万年历的设计

    文末下载完整资料 基于51单片机的电子万年历的设计 摘 要 电子万年历是单片机系统的一个应用 xff0c 由硬件和软件相配合使用 硬件由主控器 时钟电路 温度检测电路 显示电路 键盘接口5个模块组成 主控模块用AT89C52 时钟电路用时钟
  • 基于单片机的数字钟设计

    文末下载完整资料 摘 要 基于单片机的定时和控制装置在许多行业有着广泛的应用 xff0c 而数字钟是其中最基本的 xff0c 也是最具有代表性的一个例子 在基于单片机系统的数字钟电路中 xff0c 除了基本的单片机系统和外围电路外 xff0
  • 电子设计大赛-电源电路

    文末下载完整资料 集成直流稳压电源的设计 直流稳压电源是电子设备的能源电路 xff0c 关系到整个电路设计的稳定性和可靠性 xff0c 是电路设计中非常关键的一个环节 本节重点介绍三端固定式 xff08 正 负压 xff09 集成稳压器 三
  • 总线的分类和区别

    总线分类 xff1a 点击查看原文 全套资料免费下载 xff1a 关注v x 公 众 号 xff1a 嵌入式基地 后 台 回 复 xff1a 电赛 即可获资料 回复 编程 即可获取 包括有 xff1a C C 43 43 C JAVA Py

随机推荐

  • 电子设计大赛-室内可见光定位装置

    室内可见光定位装置 1 1 设计任务 设计并制作可见光室内定位装置 xff0c 其构成示意图如图 1 所示 参赛者自行搭建不小于 80cm 80cm 80cm 的立方空间 xff08 包含顶部 底部和 3 个侧面 xff09 顶部平面放置
  • 嵌入式面试题

    首先给大家分享一个在线练习面试题的网站 xff1a 牛客网 该网站包含使用实例 应用技巧 基本知识点总结和需要注意事项 xff0c 具有参考价值 xff0c 需要的朋友可以参考一下 嵌入式面试题 点击进行练习 嵌入式面试题 第一部分 xff
  • ESP32 DIY 机器狗

    关注v x 公 众 号 xff1a 嵌入式基地 后 台 回 复 xff1a 电赛 即可获资料 回复 编程 即可获取 包括有 xff1a C C 43 43 C JAVA Python JavaScript PHP 数据库 微信小程序 人工智
  • 半桥与全桥的优缺点

    关注v x 公 众 号 xff1a 嵌入式基地 后 台 回 复 xff1a 电赛 即可获资料 回复 编程 即可获取 包括有 xff1a C C 43 43 C JAVA Python JavaScript PHP 数据库 微信小程序 人工智
  • 分享几个适合新手的C/C++开源项目

    分享几个适合新手的C C 43 43 开源项目 今天主要给大家分享一些github内适合初学者练手的c c 43 43 开源项目 所有项目均提供项目下载地址 xff0c 无法使用github的读者 xff0c 也可以在文末进行获取 如果说不
  • 树莓派VNC viewer显示cannot currently show the desktop的三种可能与解决办法(自己遇到过的)

    一 储存空间满了 var cache apt archives提示空间不够 xff0c 下线后再次连接VNC可能会黑屏 xff0c 改分辨率没有用 解决方法 xff1a sudo apt span class token operator
  • 程序员常用刷题网站分享

    1 牛客网 https www nowcoder com link pc csdncpt qrsjd c 该网站内集成了面试 题库 社群交流 课程教育 面试 招聘内推等多个模块 另外还是一个交流学习的平台 xff0c 在该网站经常会有大佬对
  • C语言刷题(二)

    作者简介 xff1a 大家好我是 xff1a 嵌入式基地 xff0c 是一名嵌入式工程师 xff0c 希望一起努力 xff0c 一起进步 xff01 x1f4c3 个人主页 xff1a 嵌入式基地 x1f525 系列专栏 xff1a 牛客网
  • 嵌入式软件工程师面试题(八)

    作者简介 xff1a 大家好我是 xff1a 嵌入式基地 xff0c 是一名嵌入式工程师 xff0c 希望一起努力 xff0c 一起进步 xff01 x1f4c3 个人主页 xff1a 嵌入式基地 x1f525 系列专栏 xff1a 嵌入式
  • 嵌入式软件工程师面试题(九)

    作者简介 xff1a 大家好我是 xff1a 嵌入式基地 xff0c 是一名嵌入式工程师 xff0c 希望一起努力 xff0c 一起进步 xff01 x1f4c3 个人主页 xff1a 嵌入式基地 x1f525 系列专栏 xff1a 嵌入式
  • 单片机、嵌入式的大神都平时浏览什么网站?

    单片机 嵌入式的大神都平时浏览什么网站 xff1f 1 基础学习 xff08 C C 43 43 xff0c Linux基础等 xff09 菜鸟教程 xff08 C语言学习 xff09 c语言中文网计算机科学网站 xff08 C语言部分 x
  • 一文读懂锁相环基本原理

    1 锁相环是什么 xff1f 锁相环电路是使一个特殊系统跟踪另外一个系统 xff0c 更确切的说是一种输出信号在频率和相位上能够与输入参考信号同步的电路 xff0c 它是模拟及数模混合电路中的一个基本的而且是非常重要的模块 2 锁相环基本理
  • 史上最全—毕业设计答辩技巧

    史上最全 毕业设计答辩技巧 一 常见问题二 答辩技巧三 论文答辩 图表穿插四 论文答辩 语流适中五 论文答辩六 论文答辩七 论文答辩八 论文答辩 紧扣主题九 论文答辩 人称使用十 完整版下载 一 常见问题 1 自己为什么选择这个课题 xff
  • 毕业设计答辩常见问题汇总

    毕设答辩常见问题汇总 1 P0 口需不需要加上拉电阻问题 xff1f 2 本课题的选课背景 意义等等 xff1f 3 数码管采用的是什么扫描方式 xff1f 4 蜂鸣器或继电器的驱动三极管为什么选用pnp型的 xff08 9012 8550
  • 【C语言】判断素数的函数

    文章目录 一 函数描述二 素数定义三 函数实现 一 函数描述 自定义一个函数 xff0c 传入一个整数n xff0c 判断是否为素数 若是返回1 xff0c 否则返回0 二 素数定义 素数又称质数 一个大于1的自然数 xff0c 除了1和它
  • 【C语言】判断闰年的函数

    文章目录 一 函数描述二 相关说明三 函数实现 一 函数描述 自定义一个函数 xff0c 传入一个年份n xff0c 判断是否为闰年 若是返回1 xff0c 否则返回0 二 相关说明 平年二月28 xff0c 闰年二月29 平年有365天
  • PX4串口驱动的三种方式:系统级操作、task/work_queue、类似PX4IO中断响应

    因为项目需要 xff0c 需在px4实现与stm32F407串口通信功能 xff0c 以200Hz双工交互数据 xff0c 所以探究了3种不同的串口驱动方式 三种方式概况参考了AcmeUavs公众号的文章 xff1a PX4 6 串口设备驱
  • 【C语言】求n的阶乘(递归法和循环法)

    文章目录 一 循环法二 递归法 一 循环法 根据阶乘的计算方法 xff1a n xff01 61 1 2 3 n xff0c 我们在一个for循环完成 n 次乘法运算 注意因为是连乘 xff0c 最终阶乘结果可能会非常大所以我们在Fac函数
  • 【C语言】数组排序方法总结

    一 冒泡排序 相邻元素两两比较 xff0c 按照要求交换位置 xff0c n个元素一共要比较n 1趟 xff0c 每趟要两两比较未排序元素个数 1次 span class token macro property span class to
  • 【C语言】通讯录实现

    通讯录功能 添加联系人信息 xff08 名字 xff0c 性别 xff0c 年龄 xff0c 电话号码 xff0c 家庭住址 xff09 输出指定联系人信息查找指定联系人信息修改指定联系人信息打印所有联系人信息对所有联系人 xff08 通过