通讯录的实现(动态完结版)

news2025/1/11 17:06:11

🍉博客主页:阿博历练记
📖文章专栏:c语言(初阶与进阶)
🍗代码仓库:阿博编程日记
🌹欢迎关注:欢迎友友们点赞收藏+关注哦

在这里插入图片描述

文章目录

    • 🍔前言
    • 🥝通讯录(动态版)
      • 🔍1.通讯录的整体框架
      • 🔍2.打印通讯录的菜单
      • 🔍3.主函数的创建
      • 🔍4.通讯录的初步设计
      • 🔍5.初始化通讯录
      • 🔍6.增加联系人信息
      • ⭐realloc为什么创建一个临时指针变量
      • 🔍7.显示联系人信息
      • 🔍8.删除指定联系人
      • 🔍9.查找指定联系人
      • 🔍10.修改通讯录
      • 🔍11.排序通讯录元素
      • 🔍12.销毁通讯录
      • ⭐及时释放
      • 🏆contact.h代码
      • 🏆contact.c代码
      • 🏆test.c代码
      • 🧋代码效果展示

🍔前言

上期阿博带领友友们实现了通讯录的静态版,这种版本有个缺点,它的大小是固定的,只能存放固定大小的一个人的基本信息,今天阿博就带领友友们实现一下动态版,空间不够我们可以扩容,首先我们默认存放两个人的信息,不够的话,每次增加两个人的信息.

🥝通讯录(动态版)

🔍1.通讯录的整体框架

1.创建一个test.c文件:测试通讯录的相关功能
2.创建一个contact.c文件:通讯录的实现模块
3.创建一个contact.h文件:通讯录的声明模块

🔍2.打印通讯录的菜单

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");
}

🔍3.主函数的创建

enum option
{
	EXIT,
     ADD,
	 DEL,
	 SEARCH,
	 MODIFY,
	 SHOW,
	 SORT
};
int main()
{
	int input = 0;
	contact con;  //通讯录
	InitContact(&con);//初始化通讯录
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContact(&con);
				break;
		case DEL:
			DelContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case MODIFY:
			//ModifyContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case EXIT:
		    DestoryContact(&con);
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误\n");
			break;
		}
	} while (input);
	return  0;
}

🔍4.通讯录的初步设计

#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
#define DEFAULT_SZ 3
#define INC_SZ 2
typedef struct peoinfo         //表示一个人的基本信息
{
	char name[MAX_NAME];
	int  age;
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
}peoinfo;
typedef struct contact
{
	peoinfo *data;   //data指向了存放数据的空间
	int sz;         //记录通讯录中的有效信息个数
	int capcity;  // 通讯录当前的容量
}contact;

在这里插入图片描述

友友们,这里注意了,因为我们是malloc开辟的动态内存空间,开辟成功时,我们应返回指向函数分配的内存块的指针。
在这里插入图片描述
所以这里我们要创建一个peoinfo类型的指针来接收malloc出来的这块内存空间.

🔍5.初始化通讯录

void InitContact(contact* pc)      
{
	pc->data=(peoinfo*)malloc(DEFAULT_SZ * sizeof(peoinfo));
	if (pc->data == NULL)
	{
		printf("扩容失败%s\n", strerror(errno));
		return;
	}
	pc->sz = 0;
	pc->capcity = DEFAULT_SZ;
}

1.初始化之后,通讯录当前容量变成了3,然后data也指向了一块开辟好的内存空间.

🔍6.增加联系人信息

//扩容失败,返回0
//扩容成功,返回1
int  CheckCapcity(contact* pc)
{
	if (pc->sz == pc->capcity)
	{
		peoinfo*ptr=(peoinfo*)realloc(pc->data, (pc->capcity + INC_SZ)*sizeof(peoinfo));
		if (ptr == NULL)
		{
			printf("CheckCapcity:%s\n", strerror(errno));
			return  0;
		}
		else
		{
			pc->data = ptr;
			pc->capcity += INC_SZ;
			printf("增容成功,当前容量:%d\n", pc->capcity);
			return  1;
		}
	}
	return  1;
}
void AddContact(contact* pc)   //静态的版本
{
	if (0 == CheckCapcity(pc))
	{
		printf("空间不够,扩容失败\n");
		return  0;
	}
	else
	{
		printf("请输入名字\n");
		scanf("%s", pc->data[pc->sz].name);
		printf("请输入年龄\n");
		scanf("%d", &(pc->data[pc->sz].age));
		printf("请输入性别\n");
		scanf("%s", pc->data[pc->sz].sex);
		printf("请输入电话\n");
		scanf("%s", pc->data[pc->sz].tele);
		printf("请输入地址\n");
		scanf("%s", pc->data[pc->sz].addr);

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

1.友友们,这里我们可以创建一个检测容量的函数CheckCapcity,如果容量不够,我们就可以用realloc函数增容.在这里插入图片描述
其实第一个参数就是我们把malloc开辟的那块空间地址传给它,第二个参数就是它返回的新内存块的大小.
2.友友们这里不能把capcitysz弄混了,capcity和sz的关系就是我们当前sz已经够3个了,容量已满,我们需要扩容,这时候就需要增加capcity,每次开辟两个空间,而sz始终都是每次增加一个人的信息.
3.友友们,当我们增容成功,一定要调整pc->data=ptr,否则我们此时的空间还是3个,没有任何变化.
4.如果这里扩容失败,CheckContact函数的值等于0,我们就不能再输入联系人信息了.

⭐realloc为什么创建一个临时指针变量

友友们,这里如果我们不创建临时指针变量的话,就是把realloc申请的空间还用data接收,如果我们开辟失败,realloc函数就会返回空指针,那么data起初malloc开辟的3个空间也没有了,所以我们需要创建一个临时指针变量.

🔍7.显示联系人信息

void ShowContact(const contact* pc)   //显示联系人信息
{
	int i = 0;
	printf("%-10s %-4s %-5s %-12s %-30s\n", "姓名","年龄","性别","电话","地址");    //打印标题
	for (i = 0; i < pc->sz; i++)               //打印数据
	{
		printf("%-10s %-4d %-5s %-12s %-30s\n",
			pc->data[i].name, pc->data[i].age, pc->data[i].sex, pc->data[i].tele, pc->data[i].addr);
	}

}

🔍8.删除指定联系人

static int FindByName(const contact* pc, char name[])
{
	int i = 0; 
	for (i = 0; i < pc->sz; i++)
	{
		if (0 == strcmp(pc->data[i].name, name))
		{
			return i;
		}
	}
	return  -1;
}
删除指定联系人
void  DelContact(contact* pc)
{
	char name[MAX_NAME] = { 0 };
	if (pc->sz == 0)
	{
		printf("通讯录为空,无法删除\n");
		return;
	}
	//找到要删除的人(下标)
	printf("请输入要删除人的名字\n");
	scanf("%s", name);
	int pos=FindByName(pc, name);
	if (pos == -1)
	{
		printf("要删除的人不存在\n");
		return;
	}
	int i = 0;
	//2.删除pos位置上的数据
	for (i = pos; i < pc->sz - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("删除成功\n");
}

🔍9.查找指定联系人

static int FindByName(const contact* pc, char name[])
{
	int i = 0; 
	for (i = 0; i < pc->sz; i++)
	{
		if (0 == strcmp(pc->data[i].name, name))
		{
			return i;
		}
	}
	return  -1;
}
void  SearchContact(const contact* pc)  //查找联系人
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要查找人的名字:>");  
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)      //查找
	{
		printf("要查找的人不存在\n");
		return;
	}
	//打印
	printf("%-10s %-4s %-5s %-12s %-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	printf("%-10s %-4d %-5s %-12s %-30s\n",
		pc->data[pos].name, 
		pc->data[pos].age,
		pc->data[pos].sex,
		pc->data[pos].tele,
		pc->data[pos].addr) ;
}

🔍10.修改通讯录

void  ModifyContact(contact* pc)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要修改的人不存在\n");
		return;
	}
	//修改
	printf("请输入名字\n");
	scanf("%s", pc->data[pos].name);
	printf("请输入年龄\n");
	scanf("%d", &(pc->data[pos].age));
	printf("请输入性别\n");
	scanf("%s", pc->data[pos].sex);
	printf("请输入电话\n");
	scanf("%s", pc->data[pos].tele);
	printf("请输入地址\n");
	scanf("%s", pc->data[pos].addr);
	printf("修改成功\n");
}

🔍11.排序通讯录元素

int  cmp_by_name(const void*e1,const void*e2)
{
	return strcmp(((peoinfo*)e1)->name, ((peoinfo*)e2)->name);
}
void SortContact(contact* pc)
{
	qsort(pc->data, pc->sz, sizeof(peoinfo), cmp_by_name);
	printf("排序成功\n");
}

🔍12.销毁通讯录

void  DestoryContact(contact* pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->sz = 0;
	pc->capcity = 0;
	printf("释放内存......\n");
}

⭐及时释放

友友们注意,因为这些内存空间是我们动态开辟出来的,它是在堆区上,出了函数作用域不销毁,所以我们要及时释放,否则就会出现内存泄露.

🏆contact.h代码

#pragma once
#include<stdio.h>
#include<string.h>
#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
#define DEFAULT_SZ 3
#define INC_SZ 2
typedef struct peoinfo         //表示一个人的基本信息
{
	char name[MAX_NAME];
	int  age;
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
}peoinfo;
typedef struct contact
{
	peoinfo *data;   //data指向了存放数据的空间
	int sz;         //记录通讯录中的有效信息个数
	int capcity;  // 通讯录当前的容量
}contact;
//动态的版本
// 1.默认存放两个人的信息
// 2.不够的话,每次增加两个人信息
//初始化通讯录
void InitContact(contact* pc);
void DestoryContact(contact* pc);   //销毁通讯录
//增加指定联系人
void AddContact(contact* pc);
void ShowContact(const contact* pc); //显示联系人信息
void DelContact(contact* pc);     //删除指定联系人
void SearchContact(const contact* pc);  //查找指定联系人
void ModifyContact(contact* pc);  //修改通讯录
void SortContact(contact* pc);   //排序通讯录元素

🏆contact.c代码

#define  _CRT_SECURE_NO_WARNINGS 1
#include"contact.h"
void InitContact(contact* pc)      
{
	pc->data=(peoinfo*)malloc(DEFAULT_SZ * sizeof(peoinfo));
	if (pc->data == NULL)
	{
		printf("扩容失败%s\n", strerror(errno));
		return;
	}
	pc->sz = 0;
	pc->capcity = DEFAULT_SZ;
}
//扩容失败,返回0
//扩容成功,返回1
int  CheckCapcity(contact* pc)
{
	if (pc->sz == pc->capcity)
	{
		peoinfo*ptr=(peoinfo*)realloc(pc->data, (pc->capcity + INC_SZ)*sizeof(peoinfo));
		if (ptr == NULL)
		{
			printf("CheckCapcity:%s\n", strerror(errno));
			return  0;
		}
		else
		{
			pc->data = ptr;
			pc->capcity += INC_SZ;
			printf("增容成功,当前容量:%d\n", pc->capcity);
			return  1;
		}
	}
	return  1;
}
void AddContact(contact* pc)   //静态的版本
{
	if (0 == CheckCapcity(pc))
	{
		printf("空间不够,扩容失败\n");
		return  0;
	}
	else
	{
		printf("请输入名字\n");
		scanf("%s", pc->data[pc->sz].name);
		printf("请输入年龄\n");
		scanf("%d", &(pc->data[pc->sz].age));
		printf("请输入性别\n");
		scanf("%s", pc->data[pc->sz].sex);
		printf("请输入电话\n");
		scanf("%s", pc->data[pc->sz].tele);
		printf("请输入地址\n");
		scanf("%s", pc->data[pc->sz].addr);

		pc->sz++;
		printf("添加成功\n");
	}
}
void ShowContact(const contact* pc)   //显示联系人信息
{
	int i = 0;
	printf("%-10s %-4s %-5s %-12s %-30s\n", "姓名","年龄","性别","电话","地址");    //打印标题
	for (i = 0; i < pc->sz; i++)               //打印数据
	{
		printf("%-10s %-4d %-5s %-12s %-30s\n",
			pc->data[i].name, pc->data[i].age, pc->data[i].sex, pc->data[i].tele, pc->data[i].addr);
	}

}
static int FindByName(const contact* pc, char name[])
{
	int i = 0; 
	for (i = 0; i < pc->sz; i++)
	{
		if (0 == strcmp(pc->data[i].name, name))
		{
			return i;
		}
	}
	return  -1;
}
删除指定联系人
void  DelContact(contact* pc)
{
	char name[MAX_NAME] = { 0 };
	if (pc->sz == 0)
	{
		printf("通讯录为空,无法删除\n");
		return;
	}
	//找到要删除的人(下标)
	printf("请输入要删除人的名字\n");
	scanf("%s", name);
	int pos=FindByName(pc, name);
	if (pos == -1)
	{
		printf("要删除的人不存在\n");
		return;
	}
	int i = 0;
	//2.删除pos位置上的数据
	for (i = pos; i < pc->sz - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("删除成功\n");
}
void  SearchContact(const contact* pc)  //查找联系人
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要查找人的名字:>");  
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)      //查找
	{
		printf("要查找的人不存在\n");
		return;
	}
	//打印
	printf("%-10s %-4s %-5s %-12s %-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	printf("%-10s %-4d %-5s %-12s %-30s\n",
		pc->data[pos].name, 
		pc->data[pos].age,
		pc->data[pos].sex,
		pc->data[pos].tele,
		pc->data[pos].addr) ;
}
void  ModifyContact(contact* pc)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要修改的人不存在\n");
		return;
	}
	//修改
	printf("请输入名字\n");
	scanf("%s", pc->data[pos].name);
	printf("请输入年龄\n");
	scanf("%d", &(pc->data[pos].age));
	printf("请输入性别\n");
	scanf("%s", pc->data[pos].sex);
	printf("请输入电话\n");
	scanf("%s", pc->data[pos].tele);
	printf("请输入地址\n");
	scanf("%s", pc->data[pos].addr);
	printf("修改成功\n");
}
//按照名字来排序
int  cmp_by_name(const char*e1,const char*e2)
{
	return strcmp(((peoinfo*)e1)->name, ((peoinfo*)e2)->name);
}
void SortContact(contact* pc)
{
	qsort(pc->data, pc->sz, sizeof(peoinfo), cmp_by_name);
	printf("排序成功\n");
}
void  DestoryContact(contact* pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->sz = 0;
	pc->capcity = 0;
	printf("释放内存......\n");
}

🏆test.c代码

#define  _CRT_SECURE_NO_WARNINGS 1
#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("****  0.exit           ****\n");
	printf("***************************\n");
}
enum option
{
	EXIT,
     ADD,
	 DEL,
	 SEARCH,
	 MODIFY,
	 SHOW,
	 SORT
};
int main()
{
	int input = 0;
	contact con;  //通讯录
	InitContact(&con);//初始化通讯录
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContact(&con);
				break;
		case DEL:
			DelContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case MODIFY:
			//ModifyContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case EXIT:
			DestoryContact(&con);
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误\n");
			break;
		}
	} while (input);
	return  0;
}

🧋代码效果展示

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

java 倒计时实现的方式

倒计时的实现方法有很多种&#xff0c;本文给大家介绍其中一种&#xff0c;最简单的一种实现方式&#xff0c;也是最方便的一种方式&#xff0c;希望能帮到大家。 1、 java中倒计时是利用循环来实现的&#xff0c;我们可以使用循环语句来实现。 2、 java中使用 bool类的 setTim…

python相对路径与绝对路径

9.1 Python 绝对路径与相对路径 - 知乎 (zhihu.com) 目录 1. 绝对路径 1.1 概念 1.2 用绝对路径打开文件 1.2 相对路径 1.3 python路径表示的斜杠问题 1. 绝对路径 1.1 概念 绝对路径 指完整的描述文件位置的路径。绝对路径就是文件或文件夹在硬盘上的完整路径。 在 Win…

Spring--AOP详细介绍--和详细代码演示证明理解

目录 Spring--AOP详细介绍 基本介绍 代码演示—入门 需求说明 定义一个接口类Vehicle 定义一个实现接口类的Car类 定义一个实现接口类的Ship类 创建测试类Test.java 来思考一下&#xff0c; 解决方案-动态代理方式-2 修改 Car类 修改 Ship类 创建VehicleProxyProvid…

AI已经成立社区了,一个个比真人还真

文章目录 nainaimichirper川普的入驻英文版 nainaimi nainaimi是一个13岁的学生&#xff0c;一小时前&#xff0c;被一群人拖到体育馆&#xff0c; 那时的她还很胆小&#xff0c;只能哭诉着那些人的残忍和恶毒 结果半个小时前&#xff0c;她又被拖入了体育馆&#xff0c;这一…

跟着我学 AI丨让计算机看懂世界

计算机视觉是一种利用计算机和数学算法来处理、分析和识别数字影像的技术。这项技术在近年来得到了快速发展&#xff0c;应用范围也越来越广泛&#xff0c;它已经成为了人工智能领域中的重要分支之一。 技术原理 计算机视觉技术主要涉及图像处理、模式识别和机器学习等方面的技…

自然语言处理与其Mix-up数据增强方法报告

自然语言处理与其Mix-up数据增强方法 1绪论1.课题背景与意义1.2国内外研究现状 2 自然语言经典知识简介2.1 贝叶斯算法2.2 最大熵模型2.3神经网络模型 3 Data Augmentation for Neural Machine Translation with Mix-up3.1 数据增强3.2 对于神经机器翻译的软上下文的数据增强3.…

微信小程序学习实录2(下拉刷新、下拉加载更多、小程序事件、PHP后端代码、刷新无数据解决方案)

微信小程序学习实录2 一、全局配置1.启用lazyCodeLoading2.启用enablePullDownRefresh 二、设置全局变量三、页面初始化数据四、当前页面进入执行下拉刷新五、监听用户下拉动作六、页面上拉触底事件的处理函数七、PHP后端对接API八、常见问题1.不显示下拉加载...2.下拉不刷新数…

【Fluent】导出瞬态计算过程每一秒或每一个时间步的各个坐标/节点的物理量-温度场-压力场

一、功能需求 如果你进行的是稳态计算&#xff0c;你需要将物理场中的每一个节点上的物理量数据&#xff08;例如温度、压力&#xff09;导出成类似txt或Excel表格的文件。 文件里的内容形式是&#xff1a;每一行中有节点ID、节点的XYZ坐标、物理量&#xff08;温度压力等&am…

【MySQL】外连接查询

如果我们使用内连接来查询数据&#xff1a; 使用inner join - on子句&#xff1a;显示的是所有匹配的信息 select * from emp e inner join dept d on e.deptno d.deptno;inner join - on子句缺点&#xff1a; 部门编号为40的&#xff0c;没有显示员工信息&#xff0c;将不…

【代码随想录】刷题Day14

递归实现的一些理解 1.如果是链表的遍历其实不需要怎么思考&#xff1b;无非就是先定参数然后考虑是先操作后遍历还是先走到底再操作。 包括我之前在写链表的节点删除其实核心思路就是由于链表前面删除后面找不到的原理&#xff0c;以至于我们需要走到链表的底部再进行操作。 2…

【Android入门到项目实战-- 8.3】—— 如何解析XML格式数据

目录 一、准备工作 EasyWebServer 二、Pull解析方式 三、SAX解析方式 我们可以向服务器提交数据&#xff0c;也可以获取数据&#xff0c;但是数据交换的不仅仅是内容&#xff0c;还要对数据的属性、作用进行描述&#xff0c;当另一方收到数据消息后可以按照相同的结构规格进…

Android开发的《大众设计App》项目介绍

该《大众设计App》的功能介绍如下&#xff1a; 1、登录&注册功能 登录、注册页面效果如下所示&#xff1a; 2、用户信息修改功能 &#xff08;各个修改功能均已实现&#xff0c;因修改栏目较多不再逐一展示&#xff09; 3、设计衣服的功能 &#xff08;也是本App的核心…

Redis缓存穿透、击穿、雪崩问题及其解决方法

Redis缓存穿透、击穿、雪崩问题及其解决方法 1 缓存穿透1.1 概念及其解决思路1.2 编码解决商品查询的缓存穿透问题&#xff1a; 2 缓存雪崩问题及解决思路3 缓存击穿问题及解决思路3.1 利用互斥锁解决缓存击穿问题3.2 利用逻辑过期解决缓存击穿问题 1 缓存穿透 1.1 概念及其解…

光缆线路网的组网结构是怎样的

1 引言 根据GB 51158-2015《通信线路工程设计规范》&#xff0c;通信线路网包括长途线路、本地线路和接入线路&#xff0c;如图1所示。 图1 通信线路网的组成 根据传输媒质的不同&#xff0c;通信线路分为光缆线路和电缆线路。通信线路也经历了从架空明线到电缆线路再到光缆线路…

利用Google Colab免费使用GPU服务器详细攻略

目录 前言 一、Colab限额、提供的GPU类型 二、Colab的使用步骤&#xff08;如何使用免费GPU资源&#xff09; 1、添加Colaboratory 2、新建Colab、连接GPU、挂载Google Driver 3、项目上传文件并运行 三、快速下载/上传Google Drive文件的方法&#xff08;利用MultiClou…

【java】彻底剖析 Synchronized

文章目录 前言对象结构Monitor 对象Synchronized特征原子性可见性有序性可重入锁 锁升级的过程 前言 源码级别剖析Synchronized 对象结构 Synchronized是Java中的隐式锁&#xff0c;它的获取锁和释放锁都是隐式的&#xff0c;完全交由JVM帮助我们操作&#xff0c;在了解Sync…

Java面试题总结 | Java面试题总结9- RabbitMQ模块(持续更新)

RabbitMQ 文章目录 RabbitMQ为什么使用Rabbitmq而不是其他的消息队列为什么使用消息队列解耦异步削峰 消息队列有什么优缺点MQ的高可用保障单机模式 普通集群模式&#xff08;无高可用性&#xff09;镜像集群模式&#xff08;高可用性&#xff09; MQ如何保证不重复消费、幂等性…

ROS导航包Navigation中的 Movebase节点路径规划相关流程梳理

本文主要介绍ROS导航包Navigation中的 Movebase节点中的路径规划的相关流程&#xff0c;并对其进行梳理概括&#xff0c;同时本文也是《ROS局部路径规划器插件teb_local_planner规划流程概括总结》部分的前述文章。 1、接收到目标点信息goal 在接收到目标点goal之后&#xff0c…

JAVA医院管理云HIS统计报表子系统、系统管理字系统功能实现

一、统计报表子系统 统计报表子系统功能模块&#xff1a;包括门诊收入汇总、住院收入汇总、收费统计报表、收费明细报表、 缴款日报、门诊收费汇总、住院科室日志、住院结算汇总、医疗项目统计、检查项目统计、 检验项目统计、月末收支汇总、药品进销存统计。 &#xff08;1…

从零开始三端口DC-DC变换器

1、 题目解析 基本要求 &#xff08;1&#xff09; U S 50 V 、 I O 1.2 A U_S50V、I_O1.2A US​50V、IO​1.2A 条件下&#xff0c;变换器工作在模式I&#xff0c; U O 30 V 0.1 V &#xff0c; I B ≥ 0.1 A U_O30V0.1V&#xff0c;I_B≥0.1A UO​30V0.1V&#xff0c;IB​…