二叉堆(Binary Heap)

news2024/11/23 10:54:06

二叉堆(Binary Heap)

  • 二叉堆概述
  • 优先级队列

二叉堆(Binary Heap)其实比较简单,但却非常有用,常见的应用二叉堆排序和优先级队列。本文将介绍二叉堆的基本性质、基本操作和二叉堆在优先级队列上的应用。

二叉堆概述

二叉堆其实就是一个完全二叉树(完全二叉树的介绍见二叉树(Binary Tree):先序遍历、中序遍历、后序遍历和层次遍历),只不过用数组存储形式。
在这里插入图片描述

这里特意保留arr[0]不用,这样可以方便地计算父亲和儿子节点的index。因此,arr[1]是二叉树的根节点。二叉堆分为最大堆和最小堆两种:

  • 最大堆:每个节点的值都大于等于它的两个子节点。
  • 最小堆:每个节点的值都小于等于它的两个子节点。

显然,最大堆最大的元素就是其堆顶arr[1],最小堆最小的元素就是其堆顶arr[1]。

优先级队列

优先级队列(Priority Queue)在插入或者删除元素后,会自动完成顺序更新,这种顺序的更新是使用了二叉堆来完成的。优先级队列最重要的操作就是插入和删除(使用最大堆就是delMax,使用最小堆就是delMin),主要接口如下:

template <typename T>  // 元素类型
class MaxPriQueue {
  private:
    T* pq
    int num = 0;
    
  public:
    MaxPriQueue(int n) {
      pq = new T[n + 1];
    }
    void insert(T value);
    T delMax()
  private:
    int parent(int root) {
      return root / 2;
    }
    int left(int root) {
      return root * 2;
    }
    int right(int root) {
      return root * 2 + 1;
    }
    T max() {
      return pq[1];
    }
    void swap(int i, int j) {
      T tmp = pq[i];
      pq[i] = pq[j];
      pq[j] = tmp;
    }
    bool less(int i, int j) {
      return pq[i] < pq[j];
    }
    void swim(int x);  // 上浮第x个元素
    void sink(int x);  // 下沉第x个元素
}

二叉堆最重要的两个操作就是上浮(swim)和下沉(sink),二叉堆就是靠这两个方法维护其基本性质。这里都以以最大堆为例,最小堆情况是类似的。当我们在二叉堆里插入或者删除元素时候,肯定会破坏二叉堆的性质。当某个父节点元素小于其子节点元素时,就需要将其与子节点进行交换,也就是下沉。反之,就需要上浮。

节点的上浮可能需要多次才能到达正确的位置,代码如下:

void swim(int x) {
  while(x > 1 && less(parent(x), x)) {
    sawp(parent(x), x);
    x = parent(x);
  }
}

节点A下沉需要比较两个子节点,只要比其中之一小就需要下沉,并且将较大的子节点与A节点交换。代码如下:

void sink(int x) {
  while(left(x) <= num) {
    int maxIndex = left(x);
    if(right(x) <= num && less(maxIndex, right(x))) {
      maxIndex = right(x);
    }
    if(less(maxIndex, x)) break;
    swap(x, maxIndex);
    x = maxIndex;
  }
}

有了上浮和下沉两个方法,优先级队列的insert和delMax就比较容易了。insert就是将元素插入数组尾部,然后再执行上浮到正确位置即可。

void insert(T value) {
  num++;
  pq[num] = value;
  swim(num);
}

delMax只需要将堆顶元素A和堆底元素B交换,然后删除堆底的A,再将堆顶的B下沉到正确位置即可。

T delMax() {
  T max = pq[1];
  swap(1, num);
  pq[num] = nullptr;
  num--;
  sink(1);
  return max;
}

二叉堆就是一棵完全二叉树,非常适合使用数组存储,而二叉堆的性质主要就是靠上浮和下沉来维护。

C++标准库有优先级队列可直接使用:

template<
    class T,
    class Container = std::vector<T>,
    class Compare = std::less<typename Container::value_type>
> class priority_queue;

详情戳这里。优先级队列的应用可以刷一下力扣合并 K 个升序链表。

参考:

  • https://labuladong.github.io/algo/di-yi-zhan-da78c/shou-ba-sh-daeca/er-cha-dui-1a386/
  • https://en.cppreference.com/w/cpp/container/priority_queue

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

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

相关文章

abc 283E 经典dp

题意&#xff1a;https://www.luogu.com.cn/problem/AT_abc283_e 思路&#xff1a;非常经典的dp&#xff0c;设为前i行第i行是否反转和第i1行是否反转。 /*keep on going and never give up*/ #include<cstdio> #include<iostream> #include<queue> #inclu…

【Mysql】基础篇:DDL (data definition language) 总结

博主简介&#xff1a;努力学习的大一在校计算机专业学生&#xff0c;热爱学习和创作。目前在学习和分享&#xff1a;数据结构、Go&#xff0c;Java等相关知识。博主主页&#xff1a; 是瑶瑶子啦所属专栏: Mysql从入门到精通近期目标&#xff1a;写好专栏的每一篇文章 目录 一、…

pytest自动化框架之allure测试报告的用例描述设置

allure测试报告的用例描述相关方法&#xff1b;如下图 allure标记用例级别severity 在做自动化测试的过程中&#xff0c;测试用例越来越多的时候&#xff0c;如果执行一轮测试发现了几个测试不通过&#xff0c;我们也希望能快速统计出缺陷的等级。 pytest结合allure框架可以对…

Linux基础指令大讲解

文章目录 1.linux的历史2.如何安装linux3.如何创建和删除用户4.目录&#xff0c;文件&#xff0c;文件夹区别5.pwd6.ls6.1基础语法6.2常用选项 7.alias 和 which的配套使用7.1基础格式7.2解释7.3与其关联的一些命令 8.cd8.1相对路径和绝对路径8.2运行可执行程序的方法8.3cd的基…

Java学习(韩顺平670-675)

目录 在我看来&#xff0c;我们在Java需要不断封包的原因就是为了使用不同类库中自己所需的方法&#xff0c;我们才要进行封装的&#xff0c;你怎么看&#xff1f; 你好&#xff0c;什么叫做关闭外层流 关于字符流要求 服务端/客户端 TCP网络通信编程//●//应用案例4 TCPFil…

dbForge Edge 4in1 Enterprise Edition Crack

单一解决方案中的四个数据库 IDE dbForge Edge 赋予自己开发和管理 SQL Server、MySQL、Oracle 和 PostgreSQL 数据库的广泛能力 概述 dbForge Edge&#xff1a;您的终极多数据库解决方案 让我们来看看。您需要处理多个数据库管理系统。同时&#xff0c;您希望能够灵活有效地…

好用的项目管理软件的具体功能有哪些

随着企业规模不断的扩大&#xff0c;项目管理往往会面临更多的挑战与难题&#xff0c;最常见的会出现以下几个问题&#xff1a;资源消耗失控&#xff0c;而项目部门和相关部门之间沟通越来越困难&#xff1b;团队凝聚力下降、项目进度难以把控&#xff0c;项目成本几乎失控&…

书接上回,接着研究当今世上缓存第一人

书接上回&#xff0c;接着研究当今世上缓存第一人&#xff0c;记得留下的关注&#xff0c;收藏起来以后观看。 五 三大特殊数据类型 1 geospatial&#xff08;地理位置&#xff09; 1.geospatial将指定的地理空间位置&#xff08;纬度、经度、名称&#xff09;添加到指定的ke…

国民技术N32G430开发笔记(12)- IAP升级 Settings区域数据初始化

IAP升级 Settings区域数据初始化 1、假如&#xff0c;有两个产品&#xff0c;A产品跟B产品&#xff0c;硬件都一样&#xff0c;要求一个软件里的board_name为N32G430C8L7_STB_A&#xff0c;另一个软件里的board_name为N32G430C8L7_STB_B。 那我们如何在不改boot程序跟App程序的…

Matlab App Designer制作一个UART的GUI总结

目录 前言 读取串口数据速度慢 Axes响应慢 总结 前言 最近尝试了一下用Matlab App Designer制作一个上位机通信软件&#xff0c;专门用于读取电机控制里面的相电流&#xff0c;角度等高实时性的数据&#xff0c;已经在VOFA里面实现了。STM32 VOFA串口调试助手提升电机控制调试…

拷贝构造函数和运算符重载

文章目录 拷贝构造函数特点分析拷贝构造函数情景 赋值运算符重载运算符重载operator<运算符重载 赋值运算符前置和后置重载 拷贝构造函数 在创建对象的时候&#xff0c;是不是存在一种函数&#xff0c;使得能创建一个于已经存在的对象一模一样的新对象&#xff0c;那么接下…

Kotlin注解

文章参考 1、概念 注解可以对 类、函数、函数参数、属性等 做标注 注解信息可用于 源码级、编译期、运行时 示例 Retention(AnnotationRetention.RUNTIME) //作用于运行时 Target(AnnotationTarget.CLASS) //限定作用于类 annotation class Api(val url: String)/*** Rete…

Nachos系统的上下文切换

Fork调用创建进程 在实验1中通过gdb调试初步熟悉了Nahcos上下文切换的基本流程&#xff0c;但这个过程还不够清晰&#xff0c;通过源码阅读进一步了解这个过程。 在实验1中通过执行Threadtest&#xff0c;Fork创建子进程&#xff0c;并传入SimpleThread执行currentThread->…

vulnhub靶场之Wayne Manor

1.信息收集 探测存活主机&#xff0c;发现192.168.239.174存活 对主机进行端口扫描&#xff0c;发现存在端口21、22、80&#xff1b;且21端口状态是filtered。 打开浏览器访问http://192.168.239.174&#xff0c;未发现可疑&#xff0c;进行目录扫描只发现robots.txt。 什…

集合详解之(七)泛型

文章目录 &#x1f412;个人主页&#x1f3c5;JavaSE系列专栏&#x1f4d6;前言&#xff1a;&#x1f380;泛型的由来--参数化类型&#x1fa80;元组的介绍 ( int...数组名x )&#x1fa84;类型通配符&#x1f387;使用泛型的注意事项&#x1f3c5;子类继承泛型类【两种情况】 …

Transformers 发展一览

动动发财的小手&#xff0c;点个赞吧&#xff01; Transformers 研究概览 1. 介绍 近年来&#xff0c;深度学习的研究步伐显着加快&#xff0c;因此越来越难以跟上所有最新发展。尽管如此&#xff0c;有一个特定的研究方向因其在自然语言处理、计算机视觉和音频处理等多个领域取…

CLion开发工具 | 04 - CLion内置工具和插件

专栏介绍 一、CLion内置工具 1. SSH终端工具 填写ssh远程连接信息&#xff1a; 连接后在Terminal栏即可使用&#xff1a; 2. HTTP请求 填写http请求内容&#xff0c;并发起请求&#xff0c;方便的一批&#xff1a; 二、插件 1. 管理已安装的插件 2. 简体中文插件 3. 主题…

React 环境搭建,并打包到服务器

一. 安装node.js brew install node 二. 创建react app npx create-react-app my-app cd my-app npm start 默认使用3000端口&#xff0c;本地3000端口被占用&#xff0c;修改/node_modules/react-scripts/scripts/start.js中的端口号 // 这是start.js部分源码 const DEFAU…

【前端客栈】使用CSS实现畅销书排行榜页面

&#x1f4ec;&#x1f4eb;hello&#xff0c;各位小伙伴们&#xff0c;我是小浪。大家都知道&#xff0c;我最近是在更新各大厂的软件测试开发的面试真题&#xff0c;也是得到了很大的反馈和好评&#xff0c;几位小伙伴也是成功找到了测开的实习&#xff0c;非常不错。如果能前…

移除链表元素

☃️个人主页&#xff1a;fighting小泽 &#x1f338;作者简介&#xff1a;目前正在学习C语言和数据结构 &#x1f33c;博客专栏&#xff1a;leetcode练习题 &#x1f3f5;️欢迎关注&#xff1a;评论&#x1f44a;&#x1f3fb;点赞&#x1f44d;&#x1f3fb;留言&#x1f4a…