数据结构-(二叉)堆的模板(可用功能:建堆,删除最值,插入元素)

本文介绍了二叉堆的特性,包括完全二叉树的定义和大根堆、小根堆的概念。堆的常见操作包括建堆、删除最值和插入元素。建堆通过向上调整实现,而添加元素和删除最值则分别通过向上调整和向下调整来完成。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

堆是一种特殊的数据结构,它有两个特性:

  1. 底层实现结构是完全二叉树(定义:当且仅当一棵树每一个结点都与深度为K的满二叉树中编号从1至n的结点一一对应时称之为完全二叉树。)
  2. 如果是大根堆,那么保证每个节点的值都比父节点小,如果是小根堆,那么保证每个节点的值都比父节点大

建堆就是把一堆数据塞进堆里,因为数据很可能是无序的,所以说为了把一堆杂乱无章的数据整合成上面的样子, 就需要一些特别的操作,有两种调整模式:

  1. 向上调整。也就是对于一个节点要和它的父节点比较大小,若需要调整则和父节点交换,然后把查询位置更新到父节点的位置,继续迭代查询,终止条件为当前位置是根节点的位置。
  2. 向下调整。也就是对于每层的有子结点的父节点来说,向下查询那个较大的子结点的值是不是大于本父节点(因为有两个子节点),如果是,就交换,然后迭代向下查询,终止条件是当前节点不存在子节点。

添加数据:就是在根末(数组尾部)添加一个元素,然后从此元素的位置进行向上调整即可

删除最大值/最小值 (分别对应大根堆和小根堆):只需将根节点和根末节点(数组尾部元素)进行交换,然后将容量减一抛出根节点,再从根节点位置进行向下调整即可。

#include<pch.h>
#include <iostream>
#include <cstdio>
#include <bits/stdc++.h>
#include <map>
#include <algorithm>
#include <stack>
#include <iomanip>
#include <cstring>
#include <cmath>
#define DETERMINATION main
#define lldin(a) scanf_s("%lld", &a)
#define println(a) printf("%lld\n", a)
#define reset(a, b) memset(a, b, sizeof(a))
#define debug cout<<"procedures above are available"<<endl;
using namespace std;
const int INF = 0x3f3f3f3f;
const double PI = acos(-1);
typedef long long ll;
typedef unsigned long long ull;
typedef long double ld;
const int mod = 1000000007;
const int tool_const = 19991126;
const int tool_const2 = 33;
inline ll lldcin()
{
	ll tmp = 0, si = 1;
	char c;
	c = getchar();
	while (c > '9' || c < '0')
	{
		if (c == '-')
			si = -1;
		c = getchar();
	}
	while (c >= '0' && c <= '9')
	{
		tmp = tmp * 10 + c - '0';
		c = getchar();
	}
	return si * tmp;
}
/*Maintain your determination.Nobody knows the magnificent landscape 
at his destination before the arrival with stumble.*/
/**Last Remote**/
ll Heap[12];
ll Volume = 10;
void DownwardAdjustment(ll CurrentPosition,ll Limit)
{
	while (CurrentPosition * 2 <= Limit)
	{
		ll downstair = CurrentPosition * 2;
		if (downstair + 1 <= Limit && Heap[downstair + 1] > Heap[downstair])
			downstair++;
		if (Heap[CurrentPosition] < Heap[downstair])
		{
			swap(Heap[CurrentPosition], Heap[downstair]);
			CurrentPosition = downstair;
		}
		else
			break;
	}
}
void construction(ll Limit)
{
	for (int i = Limit / 2; i >= 1; i--)
		DownwardAdjustment(i, Limit);
}
void UpwardAdjustment(ll CurrentPosition)
{
	while (CurrentPosition > 1 && Heap[CurrentPosition] > Heap[CurrentPosition / 2])
	{
		swap(Heap[CurrentPosition], Heap[CurrentPosition / 2]);
		CurrentPosition /= 2;
	}
}
void Addition(ll Element)
{
	Heap[++Volume] = Element;
	UpwardAdjustment(Volume);
}
void EraseRoot(ll Root,ll &Limit)
{
	swap(Heap[Root], Heap[Limit]);
	Limit--;
	DownwardAdjustment(Root, Limit);
}
int DETERMINATION()
{
	ios::sync_with_stdio(false);
	cin.tie(0), cout.tie(0);
	for (int i = 1; i <= Volume; i++)
		Heap[i] = rand() % 100+1;
	for (int i : Heap)
		if(i!=0)
			cout << i << " ";
	cout << endl;
	debug;
	construction(Volume);
	ll n;
	cin >> n;
	Addition(n);
	while (Volume >= 1)
	{
		cout << Heap[1] << endl;
		EraseRoot(1, Volume);
	}
	return 0;
}

 

### 如何使用 Keil5 Hex 文件 对于仅拥有已编译好的 hex 文件而无源文件的情况,在 Keil V5 平台上直接hex 文件至单片机(如华大单片机)需采取特定的方法,因为直接调用该平台进行此类操作不可行[^1]。 #### 设置 Output 路径 进入 Keil 的 output 设置界面,指定要录的 hex 文件的具体位置。确保在路径输入框中填完整的 hex 文件名称并附带 `.hex` 扩展名;缺少此扩展名可能导致系统继续尝试录先前编译的结果而非所选的 hex 文件[^3]。 #### 配置 Flash 工具选项 针对不同类型的微控制器(MCU),可能还需调整 flash 下载工具的相关配置参数以匹配目标设备的要求。这一步骤通常涉及选择合适的编程算法以及设定通信接口等细节[^2]。 #### 启动下载过程 完成上述准备工作之后,可以通过点击调试窗口内的 “Download” 或者快捷菜单里的相应命令来启动实际的程序入流程。如果一切顺利的话,软件会自动连接硬件并将选定的 hex 数据传输到 MCU 中存储起来[^4]。 ```python # Python 示例代码用于说明自动化脚本概念 (并非真实实现) def download_hex_to_mcu(hex_file_path, mcu_type): """ 自定义函数模拟将 HEX 文件下载到指定型号的 MCU 上 参数: hex_file_path -- 完整路径字符串指向待上传的 .hex 文件 mcu_type -- 字符串表示的目标单片机类型标识符 返回值: 成功则返回 True ,失败抛出异常信息 """ try: configure_output_settings(hex_file_path) # 设定输出设置 select_flash_tool(mcu_type) # 挑选适合的闪存工具 execute_download_command() # 发送下载指令 return True # 表明成功结束 except Exception as e: raise RuntimeError(f"Failed to upload {hex_file_path}: {e}") ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值