0x17 二叉堆

news2025/1/15 17:45:38

0x17 二叉堆

二叉堆是一种支持插入、删除、查询最值的数据结构。它其实是一种满足“堆性质”的完全二叉树,树上的每一个节点带有一个权值。若树中的任意一个节点的权值都小于等于其父节点的权值,则称该二叉树满足“大根堆性质”,称其为“大根堆”。若树中的任意一个节点的权值都大于等于其父节点的权值,则称该二叉树满足“小根堆性质”,称其为“小根堆”。

根据完全二叉树性质,我们可以采取层次序列储存方式,直接用一个数组保存二叉堆。层次序列存储方式,就是逐层从左往右为树中的节点依次编号,把此编号作为节点在数组中储存的位置(下标)。在这种储存方式中,父节点编号等于子节点编号除以2,左子节点编号等于父节点编号乘2,右子节点编号等于父节点编号乘2加1,如下图所示:

在这里插入图片描述

我们以大根堆为例,探讨堆支持的常见几种操作。

Insert:

insert(val)操作向二叉堆中插入一个带有权值val的新节点。我们吧这个新节点直接放在储存二叉堆的数组末尾,然后通过交换的方式向上调整,直至满足堆性质。其时间复杂度为堆的深度,即 O ( l o g N ) O(logN) O(logN)

在这里插入图片描述

int heap[SIZE],n;
void up(int p) //向上调整
{
    while(p>1)
    {
        if(heap[p]>heap[p/2])
        {
            swap(heap[p],heap[p/2]);
            p/=2;
        }
        else
            break;
    }
}
void insert(int val)
{
    heap[++n]=val;
    up(n);
}

GetTop:

GetTop返回二叉堆的堆顶权值,即最大值heap[1],时间复杂度为 O ( 1 ) O(1) O(1)

Extract:

Extract操作把堆顶元素从二叉堆中移除。我们把堆顶heap[1]与储存数组末尾的节点heap[n]交换,然后移除数组末尾节点(令n减少1),最后把堆顶通过交换的方式向下调整,直至满足堆性质。其时间复杂度为堆的深度,即 O ( l o g N ) O(logN) O(logN)

在这里插入图片描述

void down(int p)
{
    int s=p*2; //p的左子节点
    while(s<=n)
    {
        if(s<n&&heap[s]<heap[s+1])
            s++;
        if(heap[s]>heap[p])
        {
            swap(heap[s],heap[p]);
            p=s,s=p*2;
        }
        else
            break;
    }
}

void Extract()
{
    heap[1]=heap[n--];
    down(1);
}

Remove:

Remove(p)操作把存储在数组下标p位置的节点从二叉堆中删除。与Extract相类似,我们把heap[p]heap[n]交换,然后令n减少1。注意此时heap[p]既有可能需要向下调整,也有可能需要向上调整,需要分别检查和处理。时间复杂度为 O ( l o g N ) O(logN) O(logN)

void remove(int k)
{
    heap[k]=heap[n--];
    up(k),down(k);
}

C++ STL中的priority_queue(优先队列)为实现了一个大根堆,支持push(Insert)top(GetTop)pop(Extract)操作,不支持Remove操作,详细用法参见第0x71节。

1.Huffman

考虑这样一个问题:构造一棵包含 n n n个叶子结点的 k k k叉树,其中第 i i i个叶子结点带有权值 w i w_i wi,要求最小化 ∑ w i ∗ l i \sum w_i*l_i wili,其中 l i l_i li表示第 i i i个叶子结点到根节点的距离。该问题被称为 k k kHuffman树(哈夫曼树)。

为了最小化 ∑ w i ∗ l i \sum w_i*l_i wili,应该让权值大的叶子结点的深度尽可能小。 k = 2 k=2 k=2,我们很容易想到用下面这个贪心算法来求出二叉Huffman树。

1.建立一个小根堆,插入这 n n n个叶子结点的权值。

2.从堆中取出最小的两个权值 w 1 w_1 w1 w 2 w_2 w2,令 a n s + = w 1 + w 2 ans+=w_1+w_2 ans+=w1+w2

3.建立一个权值为 w 1 + w 2 w_1+w_2 w1+w2的树节点 p p p,令 p p p成为权值 w 1 w_1 w1 w 2 w_2 w2的树节点的父亲。

4.在堆中插入权值 w 1 + w 2 w_1+w_2 w1+w2

5.重复第2~4步,直至堆的大小为1。

最后,由所有新建的 p p p与原来的叶子结点构成的树就是Huffman树,变量ans就是 ∑ w i ∗ l i \sum w_i*l_i wili的最小值。

在这里插入图片描述

对于 k ( k > 2 ) k(k>2) k(k>2)Huffman树的求解,直观的想法是在上述贪心算法的基础上,改为每次从堆中取出最小的k个权值。然而仔细思考可以发现,如果在执行最后一轮循环时,堆的大小为 2 ∼ k − 1 2\sim k-1 2k1之间(不足以取出k个),那么整个Huffman树的根的子节点的个数就小于k。这显然不是最优解——我们任取Huffman树中一个深度最大的节点,把它改为根的子节点,就会使 ∑ w i ∗ l i \sum w_i*l_i wili变小。

为此我们可以添加一些权值为0的叶子结点,使叶子结点的个数满足 ( n − 1 )   m o d   ( k − 1 ) = 0 (n-1)\bmod (k-1)=0 (n1)mod(k1)=0,然后“每次从堆中取出最小的k个权值”的贪心想法就是正确的。

在这里插入图片描述

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

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

相关文章

olap/clickhouse-编译器优化与向量化

本文主要结合15721和clickhouse源码来聊聊向量化&#xff0c;正好我最近也在用Eigen做算子加速&#xff0c;了解下还是有好处的。 提示编译器 提示编译器而不是复杂化简单的代码 什么时候使用汇编&#xff0c;什么时候使用SIMD&#xff1f;下面有几个基本原则&#xff1a; …

一种缩小数据之间差距的算法

先上代码&#xff1a; /** * 缩小数据之间的差距&#xff0c;但是大小关系不变的方法* param {Array} features */function minMaxData(data) {for (let i 0; i < data.length; i) {const f data[i];const x f[1];const yf[2];//此处5根据实际情况设置const y2 Math.pow(…

VLAN 详解一(VLAN 基本原理及 VLAN 划分原则)

VLAN 详解一&#xff08;VLAN 基本原理及 VLAN 划分原则&#xff09; 在早期的交换网络中&#xff0c;网络中只有 PC、终端和交换机&#xff0c;当某台主机发送一个广播帧或未知单播帧时&#xff0c;该数据帧会被泛洪&#xff0c;甚至传递到整个广播域。而广播域越大&#xff…

myBatis-plus自动填充插件

在 MyBatis-Plus 3.x 中&#xff0c;自动填充的插件方式发生了变化。现在推荐使用 MetaObjectHandler 接口的实现类来定义字段的填充逻辑。以下是使用 MyBatis-Plus 3.x 自动填充的基本步骤&#xff1a; 1.基本配置 1.1添加 Maven 依赖&#xff1a; 确保你的 Maven 依赖中使…

数据结构之<图>的介绍

图&#xff08;Graph&#xff09;的概念&#xff1a; 在数据结构中&#xff0c;图是由节点&#xff08;顶点&#xff09;和边组成的非线性数据结构。图用于表示不同对象之间的关系&#xff0c;其中节点表示对象&#xff0c;边表示对象之间的连接或关系。 1.图的基本组成元素&a…

Jenkins----基于 CentOS 或 Docker 安装部署Jenkins并完成基础配置

查看原文 文章目录 基于 CentOS7 系统部署 Jenkins 环境基于 Docker 安装部署 Jenkins环境配置 Jenkins 中文模式配置用户名密码形式的 Jenkins 凭据配置 ssh 私钥形式的 Jenkins 凭据配置 Jenkins 执行任务的节点 基于 CentOS7 系统部署 Jenkins 环境 &#xff08;1&#xff…

广东高院严惩“套路贷”犯罪,保护校园安全

近日&#xff0c;广东高院发布了一批依法严惩“套路贷”犯罪的典型案例&#xff0c;其中一起涉及在校学生的“套路贷”案件引起了广泛关注。 这起案件中&#xff0c;张某等人针对在校大学生开展无抵押高息短期借款“套路贷”业务&#xff0c;通过频繁威胁恐吓、借新还旧、转单…

【计算机视觉--解耦视频分割跟踪任何物体】

UIUC&Adobe开源|无需监督&#xff0c;使用解耦视频分割跟踪任何物体&#xff01;视频分割的训练数据往往昂贵且需要大量的标注工作。这限制了将端到端算法扩展到新的视频分割任务&#xff0c;特别是在大词汇量的情况下。为了在不为每个个别任务训练视频数据的情况下实现“跟…

NtripShare Mos监测平台边缘计算终端与自动优化平差算法

忙忙乎乎23年又要过去了&#xff0c;回头看今年做的事&#xff0c;只有两件事值得一提&#xff1a; 1、自动化监测边缘计算终端&#xff1b; 2、自动优化平差算法。 自动化监测边缘计算终端 终端采用全国产硬件方案终端支持全站仪供电控制终端支持远程控制终端支持数据缓存技…

力扣刷题-二叉树-二叉树的所有路径

257 二叉树的所有路径 给定一个二叉树&#xff0c;返回所有从根节点到叶子节点的路径。 说明: 叶子节点是指没有子节点的节点。 示例: 思路 参考&#xff1a; https://www.programmercarl.com/0257.%E4%BA%8C%E5%8F%89%E6%A0%91%E7%9A%84%E6%89%80%E6%9C%89%E8%B7%AF%E5%BE…

嵌入式开发中的总线与时钟

总线 AHB总线 AHB的全称是"Advanced High-performance Bus",中文翻译就是"高级高性能总线"。这是一种在计算机系统中用于连接不同硬件组件的总线架构,它可以帮助这些组件之间高效地传输数据和信息。这个总线架构通常用于处理速度较快且对性能要求较高的…

Markdown编辑器常用颜色背景指南(附颜色与代码展示,cv即可用)

目录 一.字体颜色1)常用html代码2)通过【颜色代码表】直接改写 二.字体背景颜色1)常用html代码 一.字体颜色 1)常用html代码 html代码 <font colorred> text </font> 常用颜色及其对应的十六进制和颜色名: 红色 p 红色 #FF0000 red <font color#FF0000> t…

springboot整合vue,将vue项目整合到springboot项目中

将vue项目打包后&#xff0c;与springboot项目整合。 第一步&#xff0c;使用springboot中的thymeleaf模板引擎 导入依赖 <!-- thymeleaf 模板 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-t…

Frontier ,MDPI T3系列,植物科学领域高质量期刊分级目录发布!

公众号&#xff1a;生信漫谈&#xff0c;获取最新科研信息&#xff01; Frontier &#xff0c;MDPI T3系列&#xff0c;植物科学领域高质量期刊分级目录发布&#xff01;https://mp.weixin.qq.com/s/ukbjIgdyaza7LmKmZmy5bw 2023年3月31日&#xff0c;中国科学技术大学科研部…

运筹学经典问题(二):最短路问题

问题描述 给定一个图&#xff08;有向图或无向图&#xff09; G ( V , E ) G (V, E) G(V,E)&#xff0c; V V V是图中点的集合&#xff0c; E E E是图中边的集合&#xff0c;图中每条边 ( i , j ) ∈ E (i, j) \in E (i,j)∈E都对应一个权重 c i j c_{ij} cij​&#xff08;…

第7章 排序

前言 在这一章&#xff0c;我们讨论数组元素的排序问题。为简单起见&#xff0c;假设在我们的例子中数组只包含整数&#xff0c;虽然更复杂的结构显然也是可能的。对于本章的大部分内容&#xff0c;我们还假设整个排序工作能够在主存中完成&#xff0c;因此&#xff0c;元素的个…

Redis Cluster集群搭建 三主三从

Redis包下载 Linux&#xff1a; http://download.redis.io/releases/ Mac or Windows: https://redis.io/download/ 2.下载后解压进入文件夹&#xff08;本次我的Redis版本是6.2.14版本&#xff09; /redis/redis-6.2.14 开始安装 make instarll修改配置文件复制redis.conf 6…

pve8.1版本安装及环境初始化过程记录

背景 经历pve旧版本奔溃事件&#xff0c;果断重新购买了装备&#xff0c;做个稳定的pve环境&#xff0c;旧主机用于折腾其他系统&#xff0c;硬盘一定要用好的。 1.pve8.1.3系统版本安装 下载地址 Download Proxmox software, datasheets, agreements 使用种子下载会快一点儿…

英伟达盒子 Jetson Xshell连接串口查看日志方法(串口日志、盒子日志)

文章目录 连接串口xshell连接串口信息 连接串口 接盒子上的A2、B2&#xff0c;以及接地线&#xff1a; 另外一头接上电脑&#xff08;我用的RS485转USB工具&#xff09;&#xff1a; xshell连接 协议选择SERIAL&#xff1a; 设置盒子厂商约定的端口号、波特率、数据位、停止位…

ElaticSearch 是如何建立索引的?

前面讲到了 NoSQL 数据库的应用&#xff0c;在关系型数据库和 NoSQL 数据库之外&#xff0c;还有一类非常重要的存储中间件&#xff0c;那就是文件索引。当你在电商网站搜索商品&#xff0c;或者在搜索引擎搜索资料时&#xff0c;都离不开基于文件索引的各种检索框架的支持。 …