C语言:通讯录

目录

前言

1、通讯录框架,基本介绍

2、程序实现,步骤+解释

2.1. 建立菜单

2.2. 主函数框架

2.3 前期工作

2.4 初始化通讯录函数:void InitContact(Contact* pc);

2.5 添加信息操作:void AddContact(Contact* pc);

2.6 显示通讯录信息:void ShowContact(Contact* pc);

2.7 删除通讯录信息:void DelContact(Contact* pc);

2.8 查找通讯录信息:void SearchContact(Contact* pc);

2.9 修改通讯录信息:void ModifyContact(Contact* pc);

2.10 排序通讯录信息:void SortContact(Contact* pc);

3 、效果展示

总结


前言

        此篇介绍了一个C语言的项目:通讯录。该项目是C语言进阶的实战项目,希望对你有帮助。

1、通讯录框架,基本介绍

        一个通讯录的人包含的信息:名字;年龄;性别;电话;地址。

        通讯录基本要求:

        1、能存放100人的信息

        2、能增加联系人

        3、能删除指定联系人

        4、查找联系人

        5、修改联系人

        6、排序

        7、显示联系人

2、程序实现,步骤+解释

2.1. 建立菜单

        首先我们运行程序时必须有一个菜单,指引我们如何操作:

        如上图。这样的菜单建立很简单,只需要用到printf函数打印即可:

    printf("*************************************\n");
	printf("******  1. add      2. del     ******\n");
	printf("******  3. search   4. modify  ******\n");
	printf("******  5. show     6. sort    ******\n");
	printf("******  0. exit                ******\n");
	printf("*************************************\n");

        我们将其封装成一个函数,以便我们多次调用。

void menu()
{
	printf("*************************************\n");
	printf("******  1. add      2. del     ******\n");
	printf("******  3. search   4. modify  ******\n");
	printf("******  5. show     6. sort    ******\n");
	printf("******  0. exit                ******\n");
	printf("*************************************\n");
}

2.2. 主函数框架

int main()
{
	int input = 0;
	Contact con;		//通讯录
	//初始化通讯录
	InitContact(&con);
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:		//增加
			AddContact(&con);
			break;
		case 2:		//删除
			DelContact(&con);
			break;
		case 3:		//查找
			SearchContact(&con);
			break;	
		case 4:		//修改
			ModifyContact(&con);
			break;
		case 5:		//显示
			ShowContact(&con);
			break;
		case 6:		//排序
			SortContact(&con);
			break;
		case 0:
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误,请重新输入\n");
			break;
		}
	} while(input);

	return 0;
}

        主函数的框架其实很简单,一开始时初始化一个通讯录,用到我们自己定义的函数:InitContact()。然后读取一个输入值,这个值是用户输入的,用户根据菜单打印他想要进行的操作,包括增删查改、排序和显示。此内容于菜单对应。输入1:增加信息;输入2:删除信息;输入3:查找信息;输入4:修改信息;输入5:显示信息;输入6:对信息排序;输入0:退出;其他输入:输入错误,重新输入。这样的逻辑我们用一个switch语句恰好实现,如上图。接下来我们要做的就是实现我们主函数中出现的函数的操作了。

2.3 前期工作

        定义通讯录结构体:

typedef struct Contact
{
	PeoInfo data[100];//存放人的信息
	int count;		//记录当前通讯录中实际存储人的信息个数。
}Contact;

        此结构体包括,一个存放人的信息的结构体,和一个记录通讯录存储信息个数的变量。

        PeoInfo结构体定义:

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

        存放人信息的结构体PeoInfo内容包括:姓名;年龄;性别;电话;地址。

2.4 初始化通讯录函数:void InitContact(Contact* pc);

void InitContact(Contact* pc)
{
	assert(pc);
	pc->count = 0;
	memset(pc->data, 0, sizeof(pc->data));
}

        利用memset函数,将传输过来的通讯录结构体初始化,前提包含头文件:string.h。

2.5 添加信息操作:void AddContact(Contact* pc);

void AddContact(Contact* pc)
{
	assert(pc);
	if (pc->count == MAX)
	{
		printf("通讯录已满,无法添加\n");
		return;
	}
	//
	printf("请输入名字:>");
	scanf("%s", pc->data[pc->count].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");
}

        其中MAX表示通讯录最大容量,目前设置为100。当通讯录已满时,无法加入新信息。

        通讯录未满,则要求用户对每一个量进行输入,程序将其放在对应位置,最后count++,记录通讯录信息个数的变量加一,表示通讯录又多加了一条信息。

        看效果:        

        可以看到,成功添加了该信息,只是我们测试时还用到了“5”的功能,“5”的功能是显示通讯录现有信息,后面会讲解如何实现。

2.6 显示通讯录信息:void ShowContact(Contact* pc);

        我们不按照数字顺序来,我们按照自己的逻辑顺序进行讲解,我先讲这个“5”的功能是怎么实现的,也就是如何显示通讯录:

void ShowContact(Contact* pc)
{
	assert(pc);
	printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	int i = 0;
	for (i = 0;i < pc->count;i++)
	{
		printf("%-20s\t", pc->data[i].name);
		printf("%-5d\t", pc->data[i].age);
		printf("%-5s\t", pc->data[i].sex);
		printf("%-12s\t", pc->data[i].tele);
		printf("%-30s\n", pc->data[i].addr);
	}
}

        其实逻辑很简单,就是从0开始,一直到count,也就是通讯录有的信息个数,每一个信息都全部打印一遍,用一个for循环就好了。效果就不展示了,上面已经展示了。

2.7 删除通讯录信息:void DelContact(Contact* pc);

void DelContact(Contact* pc)
{
	assert(pc);
	printf("请输入要删除的人名:>");
	char name[20] = { 0 };
	scanf("%s", name);
	int flag = FindName(pc, name);
	if (flag == -1)
	{
		printf("要删除的信息不存在\n");
		return;
	}
	else
	{
		int i = 0;
		for (i = flag;i < pc->count - 1;i++)
		{
			pc->data[i] = pc->data[i + 1];
		}
		pc->count--;
		printf("删除成功。\n");
	}

}

         删除信息的逻辑是:先通过用户输入的名字找到该信息所在的位置,然后用后面的信息覆盖掉这个信息:后面所有信息都向前挪1位。count自减1,表示通讯录存储信息少一个。

        这中间需要用到查找信息的函数:FindName。我们来看看它如何实现:

        FindName输入参数为:通讯录结构体指针;要查找的名字字符串

        返回值为:如果找到返回该数据在data中的位置,找不到返回-1。

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

	return -1;
}

        static是为了保密,这个作用就相当于给函数加锁,防止给别人看到。可以不加,函数照样实现。

        此函数逻辑为,遍历整个通讯录,找到名字和输入的一致时,返回该下表,遍历到通讯录所记录的信息个数时还未找到,返回-1。

        删除通讯录信息,效果:

2.8 查找通讯录信息:void SearchContact(Contact* pc);

void SearchContact(Contact* pc)
{
	assert(pc);
	printf("请输入要查找人的名字:");
	char name[20] = { 0 };
	scanf("%s", name);
	//查找
	int flag = FindName(pc, name);
	if (flag == -1)
	{
		printf("找不到该信息。\n");
		return;
	}
	else
	{
		printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\n",pc->data[flag].name,pc->data[flag].age, pc->data[flag].sex, pc->data[flag].tele, pc->data[flag].addr);
	}
}

        根据输入名字查找,再次运用到函数:FindName。将其返回下表对应的数据输出,如果返回-1,则打印:找不到该信息。

2.9 修改通讯录信息:void ModifyContact(Contact* pc);

void ModifyContact(Contact* pc)
{
	assert(pc);
	printf("请输入要修改的人名:");
	char name[20] = { 0 };
	scanf("%s", name);
	//查找
	int flag = FindName(pc, name);
	if (flag == -1)
	{
		printf("找不到该信息。\n");
		return;
	}
	else
	{
		printf("请输入修改后的名字:>");
		scanf("%s", pc->data[flag].name);
		printf("请输入修改后的年龄:>");
		scanf("%d", &pc->data[flag].age);
		printf("请输入修改后的性别:>");
		scanf("%s", pc->data[flag].sex);
		printf("请输入修改后的电话:>");
		scanf("%s", pc->data[flag].tele);
		printf("请输入修改后的地址:>");
		scanf("%s", pc->data[flag].addr);
		printf("修改成功\n");
	}
}

        仍旧是先通过输入的名字找到信息对应下标,要用到函数:FindName。然后再用新输入的数据覆盖老的数据,达到修改的效果。

        

2.10 排序通讯录信息:void SortContact(Contact* pc);

        通过冒泡排序,根据信息中的人名进行排序:

void SortContact(Contact* pc)
{
	assert(pc);
	//冒泡排序,按名字升序
	int i = 0;
	int j = 0;
	PeoInfo C = { 0 };
	for (i = 0;i < pc->count;i++)
	{
		for (j = 1;j < pc->count - i;j++)
		{
			if (strcmp(pc->data[j - 1].name, pc->data[j].name) > 0)
			{
				C = pc->data[j - 1];
				pc->data[j - 1] = pc->data[j];
				pc->data[j] = C;
			}
		}
	}
	printf("排序成功。");
}

        排序完成,输出:排序成功。

3 、效果展示

总结

        通过这样一个项目,可以很好地加强我们对结构体的运用,以及对C语言的掌握。希望这样的讲解对你有所帮助。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/444021.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Sublime Text 格式化Json文件 之 Pretty Json

需要使用到 Pretty Json插件。 一、安装方法 sublime 下&#xff0c;按快捷键 Comand control p&#xff0c; 输入install Package,然后回车 等几秒钟&#xff0c;加载启动进程完毕后弹出的页面中输入pretty json, 然后回车 等待几秒钟&#xff0c;可以查看Sublime 最下面的…

nginx配置支持ipv6访问,ipv4改造ipv6

一、前言 本地测试nginx部署的web系统支持ipv6地址访问。 二、本机ipv6地址 cmd ipconfig 找到IPv6地址 其中带有%号其实是临时分配得到地址 我们可以ping一下看看 另一种ping的方式 加上中括号 还有就是去掉%号 三、nginx增加配置 server块里增加 listen [::]:80; 四、测…

【SQL】指定日期的产品价格(IFNULL函数)

题目描述 leetcode题目&#xff1a;指定日期的产品价格 思路 找出所有的产品的指定的日期的价格&#xff1b;若找不到某个产品的更改日期&#xff0c;则将该产品价格设置为10。 关键点&#xff1a; if没有16号的&#xff0c;怎么找到前一个日期的&#xff1f;> 日期小…

【数字人】12、DINet | 使用形变+修复模块实现高清 talking head 生成(AAAI2023)

文章目录 一、背景二、方法2.1 deformation part2.2 inpainting part2.3 Loss 函数 三、效果3.1 数据集3.2 实现细节3.3 可视化效果 论文&#xff1a;DINet: Deformation Inpainting Network for Realistic Face Visually Dubbing on High Resolution Video 代码&#xff1a;h…

Yolov8-pose关键点检测:原创自研涨点系列篇 | 空间上下文感知模块(SCAM)结合超轻量高效动态上采样DySample

💡💡💡本文独家改进:YOLOV8-pose head创新,1)一种超轻量高效动态上采样DySample, 具有更少的参数、FLOPs,效果秒杀CAFFE和YOLOv8网络中的nn.Upsample;2)加入空间上下文感知模块(SCAM)进一步提升检测精度; 改进结构图如下: Yolov8-Pose关键点检测专栏介绍:ht…

回溯算法09-子集II(Java/子集问题的去重方法)

9.子集II 题目描述 给你一个整数数组 nums &#xff0c;其中可能包含重复元素&#xff0c;请你返回该数组所有可能的子集&#xff08;幂集&#xff09;。 解集 不能 包含重复的子集。返回的解集中&#xff0c;子集可以按 任意顺序 排列。 示例 1&#xff1a; 输入&#xf…

VMware虚拟机安装Ubuntu kylin22.04系统教程(附截图详细步骤)

一、版本信息 虚拟机产品&#xff1a;VMware Workstation 17 Pro 虚拟机版本&#xff1a;17.0.0 build-20800274 ISO映像文件&#xff1a;ubuntukylin-22.04-pro-amd64.iso 二、安装步骤 打开虚拟机&#xff0c;点击创建新的虚拟机&#xff1a; 选择自定义&#xff1a; 硬…

2024年新手视频剪辑软件推荐-6款视频剪辑软件测评

视频剪辑软件推荐 premiere premiere 直达地址:各大软件网站 说到底,还是得专业的来,虽然很多人觉得他是收费的,但是你懂的,想要免费总是会有办法的.别的不说,剪辑这块,我还是很认可这个软件,虽然我现在还是刚入门. 剪映 剪映 抖音官方推出的一款手机视频编辑剪辑应用,提供切割…

Full GC的认识、预防和定位

(/≧▽≦)/~┴┴ 嗨~我叫小奥 ✨✨✨ &#x1f440;&#x1f440;&#x1f440; 个人博客&#xff1a;小奥的博客 &#x1f44d;&#x1f44d;&#x1f44d;&#xff1a;个人CSDN ⭐️⭐️⭐️&#xff1a;传送门 &#x1f379; 本人24应届生一枚&#xff0c;技术和水平有限&am…

ChatGPT 串接到 Discord - 团队协作好助理

ChatGPT 串接到 Discord - 团队协作好助理 ChatGPT 是由 OpenAI 开发的一个强大的语言模型&#xff0c;本篇文章教你如何串接 Discord Bot &#xff0c;协助团队在工作上更加高效并促进沟通与协作。使 ChatGPT 发挥出最大的功效&#xff0c;进一步提升工作效率和团队协作能力。…

【已解决】无法删除自己上传在CSDN的资源怎么办?(2024亲测可用)

文章目录 1. 前情提要2. 实测过程3. 解决方案 1. 前情提要 我在 CSDN 上发布了一个免费资源&#xff0c;近几天却有粉丝反馈这个免费资源现在要开 VIP 才能下载&#xff0c;于是我想删除这个资源重新上传&#xff0c;但系统提示我没有权限&#xff0c;被下载过的资源无法删除&…

pytest测试框架使用基础06 fixture——parametrize

pytest.mark.parametrize 允许在测试函数或类中定义多组参数和 fixtures。 参数化场景&#xff1a; 只有测试数据和预期结果不一样&#xff0c;但操作步骤是一样的测试用例是可以用上参数化的。 创建test_cases02.py文件 示例一&#xff1a;未参数化 1.脚本代码&#xff1a; #…

Visual Studio 2022缺少项目模板的一种解决办法

检查设置 发现vs2022项目模板缺少&#xff0c;先打开vs2022&#xff0c;看看位置是否正确 缺少项目模板时处理 我在升级到&#xff1a;17.9.2时&#xff0c;在新建项目时&#xff0c;发现C#缺少“Windows窗体应用&#xff08;.Net Framework)”&#xff0c;我装了个vs201…

一个用libcurl多线程下载断言错误问题的排查

某数据下载程序&#xff0c;相同版本的代码&#xff0c;在64位系统中运行正常&#xff0c;但在32位系统中概率性出现断言错误。一旦出现&#xff0c;程序无法正常继续&#xff0c;即使重启亦不行。从年前会上领导提出要追到根&#xff0c;跟到底&#xff0c;到年后的今天&#…

力扣由浅至深 每日一题.01 两数之和

万物惊鸿&#xff0c;唯我澄明 —— 24.3.9 1. 两数之和https://leetcode.cn/problems/two-sum/ 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数&#xff0c;并返回它们的数组下标。 你可以假设每种输入只会…

Redis持久化:RDB和AOF

RDB&#xff08;Redis DataBase&#xff09; AOF&#xff08;Append Only File&#xff09; AOF重写 auto-aof-rewrite-min-size&#xff1a;如果 AOF 文件大小小于该值&#xff0c;则不会触发 AOF 重写。默认值为 64 MB;auto-aof-rewrite-percentage&#xff1a;执行 AOF 重写…

Pytorch学习 day09(简单神经网络模型的搭建)

简单神经网络模型的搭建 针对CIFAR 10数据集的神经网络模型结构如下图&#xff1a; 由于上图的结构没有给出具体的padding、stride的值&#xff0c;所以我们需要根据以下公式&#xff0c;手动推算&#xff1a; 注意&#xff1a;当stride太大时&#xff0c;padding也会变得很大…

我的 4096 创作纪念日

作者&#xff1a;明明如月学长&#xff0c; CSDN 博客专家&#xff0c;大厂高级 Java 工程师&#xff0c;《性能优化方法论》作者、《解锁大厂思维&#xff1a;剖析《阿里巴巴Java开发手册》》、《再学经典&#xff1a;《Effective Java》独家解析》专栏作者。 热门文章推荐&am…

PyQt5实现远程更新exe可执行文件

PyQt5实现远程下载更新exe可执行文件 1、实现流程 1、获取远程http地址 2、获取需要更新的exe文件 3、点击更新 4、把exe强关闭 5、下载文件 6、更新2、效果图 3、示例代码 conf.ini配置文件&#xff1a; {"http_address_edit_value": "http://xxx.com/xxx/…

python统计日志中数据从开始到结束的响应时间的最大值、最小值、平均值、中位数

应用场景&#xff1a;需要根据日志文件&#xff0c;统计出数据从开始下发到收到回复所需的时间&#xff0c;包括最大值、最小值、平均值、中位数。 日志格式如图类似&#xff0c;每一行日志开始部分就是所需要截取的时间&#xff1b;1条日记是以某些关键词作为开始&#xff0c;…