java版数据结构:堆,大根堆,小根堆

news2025/1/9 16:43:53

目录

堆的基本概念:

如何将一个二叉树调整成一个大根堆:

转成大根堆的时间复杂度

根堆中的插入,取出数据:


 

堆的基本概念:

堆是一种特殊的树形数据结构,它满足以下两个性质:

  1. 堆是一个完全二叉树,即除了最后一层外,其他层都是满的,并且最后一层的节点都尽量靠左排列。
  2. 堆中每个节点的值都大于等于(或小于等于)其子节点的值,根节点的值最大(或最小)。

根据第二个性质,堆可以分为最大堆和最小堆:

  • 最大堆:每个节点的值都大于等于其子节点的值,根节点的值最大。(又称大根堆)
  • 最小堆:每个节点的值都小于等于其子节点的值,根节点的值最小。(又称小根堆)

堆通常用于实现优先队列,可以高效地插入、删除和获取具有最高(或最低)优先级的元素。常见的堆有二叉堆、斐波那契堆等。


如何将一个二叉树调整成一个大根堆:

要将一个二叉树调整成一个大根堆,可以采用堆排序的思想,从最后一个非叶子节点开始,依次向上调整每个节点,使其满足大根堆的性质。以下是一个示例的Java代码实现:

class MaxHeap {
    public void heapify(int[] arr) {
        int n = arr.length;
        
        // 从最后一个非叶子节点开始向上调整
        for (int i = n / 2 - 1; i >= 0; i--) {
            heapifyUtil(arr, n, i);
        }
    }
    
    private void heapifyUtil(int[] arr, int n, int i) {
        int largest = i;  // 初始化最大值为当前节点
        int left = 2 * i + 1;
        int right = 2 * i + 2;
        
        // 找出当前节点、左子节点和右子节点中的最大值
        if (left < n && arr[left] > arr[largest]) {
            largest = left;
        }
        if (right < n && arr[right] > arr[largest]) {
            largest = right;
        }
        
        // 如果最大值不是当前节点,则交换位置并递归调整
        if (largest != i) {
            int temp = arr[i];
            arr[i] = arr[largest];
            arr[largest] = temp;
            
            heapifyUtil(arr, n, largest);
        }
    }
}

假设我们有一个数组 arr = {4, 10, 3, 5, 1},我们希望将其调整成一个大根堆。我们可以使用上面提供的 MaxHeap 类中的 heapify() 方法来实现。

public class Main {
    public static void main(String[] args) {
        int[] arr = {4, 10, 3, 5, 1};
        
        MaxHeap maxHeap = new MaxHeap();
        maxHeap.heapify(arr);
        
        System.out.println("大根堆数组:");
        for (int num : arr) {
            System.out.print(num + " ");
        }
    }
}

在上面的示例中,我们将数组 {4, 10, 3, 5, 1} 调整成大根堆后,输出结果应该为 {10, 5, 3, 4, 1}

调成小根堆同理。 


转成大根堆的时间复杂度

在将一个数组调整成大根堆的过程中,我们从最后一个非叶子节点开始,依次向上调整每个节点,使得每个节点都满足大根堆的性质。

  1. 从最后一个非叶子节点开始,向上遍历直到根节点。对于一个具有n个节点的完全二叉树,最后一个非叶子节点的索引为n/2 - 1。

  2. 在每个节点处,我们需要执行以下操作:

    • 比较当前节点与其左右子节点的值,找出其中最大的值。
    • 如果最大值不是当前节点的值,则交换当前节点与最大子节点的值。
    • 继续向下递归调整交换后的子节点,直到满足大根堆的性质。
  3. 因为每个节点最多需要比较和交换O(log n)次,而最后一个非叶子节点的数量是n/2,所以整个调整过程的时间复杂度为O(n)。

因此,将一个数组调整成大根堆的时间复杂度为O(n)。


根堆中的插入,取出数据:

如果您想在 Java 中实现向大根堆中插入元素和从大根堆中取出最大元素的操作,您可以修改 MaxHeap 类,添加 insert() 和 extractMax() 方法。下面是一个示例代码:

public class MaxHeap {
    public void heapify(int[] arr) {
        int n = arr.length;
        
        for (int i = n / 2 - 1; i >= 0; i--) {
            heapifyUtil(arr, n, i);
        }
    }
    
    private void heapifyUtil(int[] arr, int n, int i) {
        int largest = i;
        int left = 2 * i + 1;
        int right = 2 * i + 2;
        
        if (left < n && arr[left] > arr[largest]) {
            largest = left;
        }
        if (right < n && arr[right] > arr[largest]) {
            largest = right;
        }
        
        if (largest != i) {
            int temp = arr[i];
            arr[i] = arr[largest];
            arr[largest] = temp;
            
            heapifyUtil(arr, n, largest);
        }
    }
    
    public void insert(int[] arr, int key) {
        // 将新元素插入到数组末尾
        arr[arr.length] = key;
        
        // 从新元素的父节点开始向上调整
        int i = arr.length;
        while (i > 0 && arr[(i-1)/2] < arr[i]) {
            int temp = arr[i];
            arr[i] = arr[(i-1)/2];
            arr[(i-1)/2] = temp;
            i = (i-1)/2;
        }
    }
    
    public int extractMax(int[] arr) {
        int n = arr.length;
        if (n == 0) {
            return -1; // 堆为空
        }
        
        int max = arr[0]; // 最大元素为根节点
        arr[0] = arr[n-1]; // 将最后一个元素移到根节点
        heapifyUtil(arr, n-1, 0); // 从根节点开始向下调整
        
        return max;
    }
}

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

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

相关文章

常见物联网面试题详解

物联网一直是非常火热的行业&#xff0c;G端如智慧城市、智慧工厂、智慧园区、智慧水利、智慧矿山等行业&#xff0c;都会涉及到物联网&#xff0c;基本都是软硬一体&#xff0c;因此当面试相关企业时&#xff0c;物联网平台是面试企业重点考察的项&#xff0c;小伙伴如果从事相…

网络安全在数字时代的重要性:以近期网络安全事件为镜

在当今这个信息化爆炸的时代&#xff0c;互联网如同一张无形的网&#xff0c;将我们的生活、工作、学习紧密相连。然而&#xff0c;这张网在带来便捷的同时&#xff0c;也暗藏着无数的安全隐患。近年来&#xff0c;网络安全事件频发&#xff0c;从个人隐私泄露到企业数据被盗&a…

中国地图(2024版审图号地图)和地图变化说明

2024版shp格式审图号地图预览图&#xff1a; 新版中国地图的变化&#xff08;简述&#xff09; 国土面积的增加&#xff1a;新版中国地图显示&#xff0c;中国的国土面积从960万平方公里增加到1045万平方公里&#xff0c;增加了85万平方公里。 九段线变为十段线&#xff1a;…

如何使用 ArcGIS Pro 制作地震动画

在做某些汇报的时候&#xff0c;除了图文&#xff0c;如果有动画肯定会成为加分项&#xff0c;这里为大家介绍一下如何使用 ArcGIS Pro 制作地震动画&#xff0c;希望能对你有所帮助。 添加时间 在图层属性内&#xff0c;选择时间选项卡&#xff0c;图层时间选择每个要素具有…

每日两题 / 226. 翻转二叉树 98. 验证二叉搜索树(LeetCode热题100)

226. 翻转二叉树 - 力扣&#xff08;LeetCode&#xff09; 以后续遍历的方式交换当前节点的左右指针 /*** Definition for a binary tree node.* struct TreeNode {* int val;* TreeNode *left;* TreeNode *right;* TreeNode() : val(0), left(nullptr), ri…

自适应调节Q和R的自适应UKF(AUKF_QR)的MATLAB程序

简述 基于三维模型的UKF&#xff0c;设计一段时间的输入状态误差较大&#xff0c;此时通过对比预测的状态值与观测值的残差&#xff0c;在相应的情况下自适应调节系统协方差Q和观测协方差R&#xff0c;构成自适应无迹卡尔曼滤波&#xff08;AUKF&#xff09;&#xff0c;与传统…

你可能喜欢但也许还不知道的好用网站-搜嗖工具箱

在线工具 https://www.zxgj.cn/ 作为一个工作生活好帮手&#xff0c;在线咨询网站提供了丰富的实用功能&#xff0c;从工作中的图表制作、图片修改到生活中的各种测试、健康、娱乐、学习、理财等等涵盖面很广。 在线工具网站从界面和操作上来看对用户也很友好&#xff0c;页面…

提高Rust安装与更新的速度

一、背景 因为rust安装过程中&#xff0c;默认的下载服务器为crates.io&#xff0c;这是一个国外的服务器&#xff0c;国内用户使用时&#xff0c;下载与更新的速度非常慢&#xff0c;因此&#xff0c;我们需要使用一个国内的服务器来提高下载与更新的速度。 本文推荐使用字节…

谷歌地图商家采集在外贸客户开发中的作用和意义

谷歌地图商家采集在外贸客户开发中扮演着至关重要的角色&#xff0c;其主要作用和意义体现在以下几个方面&#xff1a; 精准定位目标市场&#xff1a;通过谷歌地图&#xff0c;外贸人员可以根据特定的行业关键词&#xff08;如“fabric stores”&#xff09;搜索目标国家或地区…

Redis加入系统服务,开机自启

vi /etc/systemd/system/redis.service i :wq #加载服务配置文件 systemctl daemon-reload #启动redis systemctl start redis #设置开机自启 systemctl enable redis #查看启动状态 systemctl status redis

11.买卖股票的最佳时机Ⅰ

文章目录 题目简介题目解答解法一&#xff1a;一次遍历代码&#xff1a;复杂度分析&#xff1a; 题目链接 大家好&#xff0c;我是晓星航。今天为大家带来的是 买卖股票的最佳时机面试题Ⅰ 相关的讲解&#xff01;&#x1f600; 题目简介 题目解答 解法一&#xff1a;一次遍历…

汗之谜语,流产之哀:肾合唤醒生命花园的璀璨绽放

在这个疾驰的时代洪流中&#xff0c;女性宛若四季更迭间绚烂绽放的花朵&#xff0c;她们在风雨的锤炼与暖阳的抚慰下&#xff0c;演绎着生命的绚烂篇章。但当这份细腻柔美的内在花园偶遇冷冽寒潮&#xff0c;诸如汗水的异常涌动与生命的意外流失&#xff0c;就如同春暖花开之际…

01面向类的讲解

指针指向类成员使用 代码&#xff1a; #include<iostream> using namespace std;class Test { public:void func() { cout << "call Test::func" << endl; }static void static_func();int ma;static int mb; //不依赖对象 }; void Test::static…

【强化学习-Mode-Free DRL】深度强化学习如何选择合适的算法?DQN、DDPG、A3C等经典算法Mode-Free DRL算法的四个核心改进方向

【强化学习-DRL】深度强化学习如何选择合适的算法&#xff1f; 引言&#xff1a;本文第一节先对DRL的脉络进行简要介绍&#xff0c;引出Mode-Free DRL。第二节对Mode-Free DRL的两种分类进行简要介绍&#xff0c;并对三种经典的DQL算法给出其交叉分类情况&#xff1b;第三节对…

基于VOLOPV2的自动驾驶环境感知系统

基于VOLOPV2的自动驾驶环境感知系统是一个复杂的系统&#xff0c;它主要负责实时检测并识别周围环境中的各种物体和信息&#xff0c;为自动驾驶车辆提供必要的感知数据。以下是对该系统的一个简要介绍&#xff1a; 环境感知是自动驾驶系统中的一个关键部分&#xff0c;它依赖于…

揭秘全网都在搜索的抖音快速涨10000粉的方法,打造真实粉丝海洋!巨量千川投流

抖音作为当下最热门的社交媒体平台之一&#xff0c;拥有数以亿计的用户。对于许多用户来说&#xff0c;快速涨粉成为了一个追逐的目标。在这篇文章中&#xff0c;我们将揭秘一些全网都在搜索的抖音快速涨粉方法&#xff0c;帮助你打造属于自己的真实粉丝海洋。巨量千川投流&…

将机械手与CodeSys中的运动学模型绑定

文章目录 1.背景介绍2.选定运动学模型3.机械手各尺寸的对应4.总结4.1.选择正确的运动学模型4.2.注意各个关节旋转的正方向。4.3.编码器零点与机械零点的偏移修正。 1.背景介绍 最近搞到了一台工业机械手&#xff0c;虽然这个机械手有自己的控制程序&#xff0c;但是我们还是想…

概述篇——计算机网络学习笔记(基于谢希仁教材与PPT)

教材用的是谢希仁的教材&#xff0c;图片来源于谢希仁老师配套的PPT 计算机网络课程PPT&#xff08;通过这个链接&#xff0c;你可以找到课程配套的ppt&#xff09; 计算机网络的定义及其特点 定义 网络 过去大众熟知的三种网络分别是提供电话、电报及传真等服务的电信网络&am…

中小企业如何通过 Tita 统一员工管理

老实说&#xff0c;我不知道如果没有Tita&#xff0c;我们会做什么&#xff1f;如何收集五个不同人的反馈、经理的反馈、员工对经理的反馈&#xff0c;并将其全部放在一个地方&#xff1f; 在没有上线 Tita 之前&#xff0c;我们不是不做&#xff0c;是因为我们做不到。 — CE…

认识下MapReduce

&#x1f50d; 什么是MapReduce&#xff1f; MapReduce是一种分布式计算模型&#xff0c;最初由Google提出&#xff0c;用于处理大规模数据集的并行计算。它将数据处理任务分解成独立的Map和Reduce两个阶段&#xff0c;以实现分布式计算和并行化处理。Map阶段负责将输入数据映…