秋招复习之堆

news/2024/5/20 6:57:39 标签: java, 算法, 数据结构, , 优先队列

目录

前言

的常用操作

的实现(大根

1.   的存储与表示

2.   访问顶元素

3.   元素入

4.   顶元素出

Top-k 问题

方法一:遍历选择

方法二:排序

方法三:

总结


前言

秋招复习之


heap」是一种满足特定条件的完全二叉树,主要可分为两种类型,如图所示。

  • 「小顶 min heap」:任意节点的值 ≤ 其子节点的值。
  • 「大顶 max heap」:任意节点的值 ≥ 其子节点的值。

作为完全二叉树的一个特例,具有以下特性。

  • 最底层节点靠左填充其他层的节点都被填满
  • 我们将二叉树的根节点称为“”,将底层最靠右的节点称为“”。
  • 对于大顶(小顶),顶元素(根节点)的值是最大(最小)的。

的常用操作

许多编程语言提供的是「优先队列 priority queue」,这是一种抽象的数据结构,定义为具有优先级排序的队列。

实际上,通常用于实现优先队列,大顶相当于元素按从大到小的顺序出队的优先队列。从使用角度来看,我们可以将“优先队列”和“”看作等价的数据结构

在实际应用中,我们可以直接使用编程语言提供的类(或优先队列类)。

类似于排序算法中的“从小到大排列”和“从大到小排列”,我们可以通过设置一个 flag 或修改 Comparator 实现“小顶”与“大顶”之间的转换。代码如下所示:

java">/* 初始化 */
// 初始化小顶
Queue<Integer> minHeap = new PriorityQueue<>();
// 初始化大顶(使用 lambda 表达式修改 Comparator 即可)
Queue<Integer> maxHeap = new PriorityQueue<>((a, b) -> b - a);

/* 元素入 */
maxHeap.offer(1);
maxHeap.offer(3);
maxHeap.offer(2);
maxHeap.offer(5);
maxHeap.offer(4);

/* 获取顶元素 */
int peek = maxHeap.peek(); // 5

/* 顶元素出 */
// 出元素会形成一个从大到小的序列
peek = maxHeap.poll(); // 5
peek = maxHeap.poll(); // 4
peek = maxHeap.poll(); // 3
peek = maxHeap.poll(); // 2
peek = maxHeap.poll(); // 1

/* 获取大小 */
int size = maxHeap.size();

/* 判断是否为空 */
boolean isEmpty = maxHeap.isEmpty();

/* 输入列表并建 */
minHeap = new PriorityQueue<>(Arrays.asList(1, 3, 2, 5, 4));
/* 初始化 */
// 初始化小顶
priority_queue<int, vector<int>, greater<int>> minHeap;
// 初始化大顶
priority_queue<int, vector<int>, less<int>> maxHeap;

/* 元素入 */
maxHeap.push(1);
maxHeap.push(3);
maxHeap.push(2);
maxHeap.push(5);
maxHeap.push(4);

/* 获取顶元素 */
int peek = maxHeap.top(); // 5

/* 顶元素出 */
// 出元素会形成一个从大到小的序列
maxHeap.pop(); // 5
maxHeap.pop(); // 4
maxHeap.pop(); // 3
maxHeap.pop(); // 2
maxHeap.pop(); // 1

/* 获取大小 */
int size = maxHeap.size();

/* 判断是否为空 */
bool isEmpty = maxHeap.empty();

/* 输入列表并建 */
vector<int> input{1, 3, 2, 5, 4};
priority_queue<int, vector<int>, greater<int>> minHeap(input.begin(), input.end());

的实现(大根

1.   的存储与表示

完全二叉树非常适合用数组来表示。由于正是一种完全二叉树,因此我们将采用数组来存储

 将索引映射公式封装成函数

java">/* 获取左子节点的索引 */
int left(int i) {
    return 2 * i + 1;
}

/* 获取右子节点的索引 */
int right(int i) {
    return 2 * i + 2;
}

/* 获取父节点的索引 */
int parent(int i) {
    return (i - 1) / 2; // 向下整除
}
/* 获取左子节点的索引 */
int left(int i) {
    return 2 * i + 1;
}

/* 获取右子节点的索引 */
int right(int i) {
    return 2 * i + 2;
}

/* 获取父节点的索引 */
int parent(int i) {
    return (i - 1) / 2; // 向下整除
}

2.   访问顶元素

java">/* 访问顶元素 */
int peek() {
    return maxHeap.get(0);
}
/* 访问顶元素 */
int peek() {
    return maxHeap[0];
}

3.   元素入

给定元素 val ,我们首先将其添加到底。添加之后,由于 val 可能大于中其他元素,的成立条件可能已被破坏,因此需要修复从插入节点到根节点的路径上的各个节点,这个操作被称为「heapify」。

考虑从入节点开始,从底至顶执行。如图所示,我们比较插入节点与其父节点的值,如果插入节点更大,则将它们交换。然后继续执行此操作,从底至顶修复中的各个节点,直至越过根节点或遇到无须交换的节点时结束。(就是一直和父比较,大就换)

设节点总数为 n ,则树的高度为 O(log⁡N) 。由此可知,化操作的循环轮数最多为  O(log⁡N) ,元素入操作的时间复杂度为  O(log⁡N) 。

java">/* 元素入 */
void push(int val) {
    // 添加节点
    maxHeap.add(val);
    // 从底至顶化
    siftUp(size() - 1);
}

/* 从节点 i 开始,从底至顶化 */
void siftUp(int i) {
    while (true) {
        // 获取节点 i 的父节点
        int p = parent(i);
        // 当“越过根节点”或“节点无须修复”时,结束化
        if (p < 0 || maxHeap.get(i) <= maxHeap.get(p))
            break;
        // 交换两节点
        swap(i, p);
        // 循环向上化
        i = p;
    }
}
/* 元素入 */
void push(int val) {
    // 添加节点
    maxHeap.push_back(val);
    // 从底至顶化
    siftUp(size() - 1);
}

/* 从节点 i 开始,从底至顶化 */
void siftUp(int i) {
    while (true) {
        // 获取节点 i 的父节点
        int p = parent(i);
        // 当“越过根节点”或“节点无须修复”时,结束化
        if (p < 0 || maxHeap[i] <= maxHeap[p])
            break;
        // 交换两节点
        swap(maxHeap[i], maxHeap[p]);
        // 循环向上化
        i = p;
    }
}

4.   顶元素出

顶元素是二叉树的根节点,即列表首元素。如果我们直接从列表中删除首元素,那么二叉树中所有节点的索引都会发生变化,这将使得后续使用化进行修复变得困难。为了尽量减少元素索引的变动,我们采用以下操作步骤。

  1. 交换顶元素与底元素(交换根节点与最右叶节点)。
  2. 交换完成后,将底从列表中删除(注意,由于已经交换,因此实际上删除的是原来的顶元素)。
  3. 从根节点开始,从顶至底执行

如图所示,“从顶至底化”的操作方向与“从底至顶化”相反,我们将根节点的值与其两个子节点的值进行比较,将最大的子节点与根节点交换。然后循环执行此操作,直到越过叶节点或遇到无须交换的节点时结束。

与元素入操作相似,顶元素出操作的时间复杂度也为 O(log⁡n) 。代码如下所示:

java">/* 元素出 */
int pop() {
    // 判空处理
    if (isEmpty())
        throw new IndexOutOfBoundsException();
    // 交换根节点与最右叶节点(交换首元素与尾元素)
    swap(0, size() - 1);
    // 删除节点
    int val = maxHeap.remove(size() - 1);
    // 从顶至底化
    siftDown(0);
    // 返回顶元素
    return val;
}

/* 从节点 i 开始,从顶至底化 */
void siftDown(int i) {
    while (true) {
        // 判断节点 i, l, r 中值最大的节点,记为 ma
        int l = left(i), r = right(i), ma = i;
        if (l < size() && maxHeap.get(l) > maxHeap.get(ma))
            ma = l;
        if (r < size() && maxHeap.get(r) > maxHeap.get(ma))
            ma = r;
        // 若节点 i 最大或索引 l, r 越界,则无须继续化,跳出
        if (ma == i)
            break;
        // 交换两节点
        swap(i, ma);
        // 循环向下化
        i = ma;
    }
}
/* 元素出 */
void pop() {
    // 判空处理
    if (isEmpty()) {
        throw out_of_range("为空");
    }
    // 交换根节点与最右叶节点(交换首元素与尾元素)
    swap(maxHeap[0], maxHeap[size() - 1]);
    // 删除节点
    maxHeap.pop_back();
    // 从顶至底化
    siftDown(0);
}

/* 从节点 i 开始,从顶至底化 */
void siftDown(int i) {
    while (true) {
        // 判断节点 i, l, r 中值最大的节点,记为 ma
        int l = left(i), r = right(i), ma = i;
        if (l < size() && maxHeap[l] > maxHeap[ma])
            ma = l;
        if (r < size() && maxHeap[r] > maxHeap[ma])
            ma = r;
        // 若节点 i 最大或索引 l, r 越界,则无须继续化,跳出
        if (ma == i)
            break;
        swap(maxHeap[i], maxHeap[ma]);
        // 循环向下化
        i = ma;
    }
}

Top-k 问题

Q:给定一个长度为 n的无序数组 nums ,请返回数组中最大的 k个元素。

方法一:遍历选择

其时间复杂度趋向于O(n2) ,非常耗时。

 当 k=n 时,可以得到完整的有序序列,此时等价于“选择排序”算法

方法二:排序

如图所示,我们可以先对数组 nums 进行排序,再返回最右边的 k 个元素,时间复杂度为 O(nlog⁡n) 。

显然,该方法“超额”完成任务了,因为我们只需找出最大的k个元素即可,而不需要排序其他元素。

方法三:

可以基于更加高效地解决 Top-k 问题,流程如图所示。

  1. 初始化一个小顶,其顶元素最小。
  2. 先将数组的前 k 个元素依次入
  3. 从第 k+1 个元素开始,若当前元素大于顶元素,则将顶元素出,并将当前元素入
  4. 遍历完成后,中保存的就是最大k 个元素。

天才!!!

java">/* 基于查找数组中最大的 k 个元素 */
Queue<Integer> topKHeap(int[] nums, int k) {
    // 初始化小顶
    Queue<Integer> heap = new PriorityQueue<Integer>();
    // 将数组的前 k 个元素入
    for (int i = 0; i < k; i++) {
        heap.offer(nums[i]);
    }
    // 从第 k+1 个元素开始,保持的长度为 k
    for (int i = k; i < nums.length; i++) {
        // 若当前元素大于顶元素,则将顶元素出、当前元素入
        if (nums[i] > heap.peek()) {
            heap.poll();
            heap.offer(nums[i]);
        }
    }
    return heap;
}
/* 基于查找数组中最大的 k 个元素 */
priority_queue<int, vector<int>, greater<int>> topKHeap(vector<int> &nums, int k) {
    // 初始化小顶
    priority_queue<int, vector<int>, greater<int>> heap;
    // 将数组的前 k 个元素入
    for (int i = 0; i < k; i++) {
        heap.push(nums[i]);
    }
    // 从第 k+1 个元素开始,保持的长度为 k
    for (int i = k; i < nums.size(); i++) {
        // 若当前元素大于顶元素,则将顶元素出、当前元素入
        if (nums[i] > heap.top()) {
            heap.pop();
            heap.push(nums[i]);
        }
    }
    return heap;
}

总共执行了 n轮入和出的最大长度为 k ,因此时间复杂度为 O(nlog⁡k) 。该方法的效率很高,当 k 较小时,时间复杂度趋向 O(n) ;当 n 较大时,时间复杂度不会超过 O(nlog⁡n) 。

另外,该方法适用于动态数据流的使用场景。在不断加入数据时,我们可以持续维护内的元素,从而实现最大的 k个元素的动态更新。


总结

  • 是一棵完全二叉树,根据成立条件可分为大顶和小顶。大(小)顶顶元素是最大(小)的。
  • 优先队列的定义是具有出队优先级的队列,通常使用来实现。
  • 的常用操作及其对应的时间复杂度包括:元素入 O(log⁡n)、顶元素出 O(log⁡n) 和访问顶元素 O(1) 等。
  • 完全二叉树非常适合用数组表示,因此我们通常使用数组来存储
  • 化操作用于维护的性质,在入和出操作中都会用到。
  • 输入 n 个元素并建的时间复杂度可以优化至 O(n) ,非常高效。
  • Top-k 是一个经典算法问题,可以使用数据结构高效解决,时间复杂度为 O(nlog⁡K) 。


http://www.niftyadmin.cn/n/5309850.html

相关文章

动态卡尺胶路检测

动态卡尺胶路检测 1. 示例效果2. 代码 1. 示例效果 使用了三个卡尺工具、一个线段工具。这种方法可以检测胶路最常见的缺陷&#xff1a;断胶和胶宽等 2. 代码 #region namespace imports using System; using System.Collections; using System.Drawing; using System.IO; …

Redis反序列化的一次问题

redis反序列化的一次问题 1. 问题描述 springbootredis不少用&#xff0c;但是一直没遇到什么问题&#xff0c;直接代码拷贝上去就用了。这次结合spring-security&#xff0c;将自定义的spring-security的UserDetails接口的实现类SecurityUser&#xff0c;反序列化取出时报错…

Phenograph聚类方法

Phenograph是一种用于深度分析的算法&#xff0c;主要评估单细胞RNA测序数据的深度。该算法的特点是能够在需要预先设定深度的情况下&#xff0c;自动地识别和分离出潜在的可能的细胞亚群。 Phenograph 的工作原理基于图论和社区发现的原理。具体步骤如下&#xff1a; 相似性计…

面向对象软件设计与分析40讲(36)软件开发过程模型之增量模型

文章目录 1 概念2 优点3 缺点4 适用范围1 概念 增量模型强调将整个项目划分为多个增量或阶段,并在每个增量中逐步构建和交付系统的功能。每个增量是对系统的一个部分进行开发、测试和交付,形成一个可用的子系统。 以下是增量过程模型的主要特点和步骤: 划分增量:根据项目…

数字IC后端设计实现之Innovus update_names和changeInstName的各种应用场景

今天吾爱IC社区小编给大家分享下数字IC后端设计实现innovus中关于update_names和changeInstName在PR中的具体使用方法。 update_names 1&#xff09;为了避免和verilog语法保留的一些关键词&#xff0c;比如input&#xff0c;output这些&#xff0c;是不允许存在叫这类名字的…

ThinkPHP5多小区物业管理系统源码(支持多小区)

基于 ThinkPHP5 Bootstrap 倾力打造的多小区物业 管理系统源码&#xff0c;操作简单&#xff0c;功能完善&#xff0c;用户体验良好 开发环境PHP7mysql 安装步骤: 1.新建数据库db_estate,还原数据db_estate.sql 2.修改配置文件&#xff1a;application/database.php 3.运…

PHP代码审计之实战审代码篇2

4. 仔细观察如下代码&#xff0c;思考代码有什么缺陷&#xff0c;可能由此引发什么样的问题&#xff1f; <?php require_once("/home/rconfig/classes/usersession.class.php"); require_once("/home/rconfig/classes/ADLog.class.php"); require_onc…

用js计算 m-n 之间所有数的和

<script>let mprompt(输入小值)let nprompt(输入大值)function fn(min,max){let sum0for(let imin;i<max;i){sumi}return sum}let allfn(m,n)console.log(和&#xff1a;${all})</script> 效果&#xff1a;