算法-堆结构和堆排序

文章目录

    • 本节大纲
    • 1. 堆结构
    • 2. 堆排序
    • 本节的代码实现合集

本节大纲

在这里插入图片描述

1. 堆结构

堆结构是为集合类里面的优先级队列来服务的
优先级队列其实就是顺序存储的二叉树结构, 我们的底层的源码里面是没有链式存储的二叉树的,二叉树的实现的细节是通过我们的数组来模拟实现的
底层的实现细节 :
数组加上一个usedSize就可以实现一个顺序存储的二叉树结构

堆的调整
堆的调整分为向上调整与向下调整, 请注意, 堆的每次调整都是基于其中
的某一个元素破坏了堆的结构, 但是其他的堆的性质没有发生改变的情况下,所以请注意, 一定是一个元素

堆的向上调整 : 该调整不用传入终止条件, 因为根节点的下标就是天然的终止条件, 从下面的条件里面我们可以发现, 根节点的父亲节点就是其本身

代码实现如下:

 /**
     * 堆的调正是整个堆操作的最重要的步骤(这里我们默认实现的是大根堆)
     * --> 根据之前我们学习堆结构时期的困惑, 不管是向上调整还是向下调整,都是因为只有某个位置的元素破坏了堆的结构, 切记是一个! ! !
     *
     * @param nums : 传入的需要调整的数组
     * @param i    : 待调整元素的下标
     *             向上调整不需要设置终点位置, 因为就是根节点自己 当 i == 0是时, (i - 1) / 2 == 0,自动跳出循环
     */
    private void heapInsert(int[] nums, int i) {
        while (nums[i] > nums[(i - 1) / 2]) {
            swap(nums, i, (i - 1) / 2);
            i = (i - 1) / 2;
        }
    }

堆的向下调整 : 这个就跟向上调整不一样了, 这个没有天然的终止条件, 可以认为的传入终止条件…

代码实现如下

/**
     * 这里跟上面的向上调整的类似的,我们都是只有其中的一个元素破坏了堆的结构(默认实现大根堆)
     *     之前我们没有理解向上调整跟向下调整的原因就是,我们忽略了只有一个元素破坏了堆结构这一条件
     * @param nums : 待调整的数组
     * @param i    : 待调整的元素的下标
     * @param end  : 结束位置
     */
    private void heapify(int[] nums, int i, int end) {
        int l = 2 * i + 1; //先把孩子下标打到左孩子的位置;
        while (l < end) {
            //下面这句话的代码逻辑是 --> 找到最大孩子的孩子下标
            int best = l + 1 < end && nums[l + 1] > nums[l] ? l + 1 : l;
            //下面这句话的代码逻辑是 --> 比较最大的孩子跟父节点
            best = nums[i] < nums[best] ? best : i;
            if (best == i) {
                break;
            }
            swap(nums, best, i);
            i = best;
            l = 2 * i + 1;
        }
    }

堆的调整的时间复杂度分析 --> 堆的调整的时间复杂度是O(LogN), 依据的是二叉树的树的高度

2. 堆排序

堆排序其实就是堆结构的应用, 原理就是(这里按照升序分析) :
传入一个数组然后建堆(可以从顶部到底部建堆, 也可以反着来)
两种建堆的时间复杂度是不一样的, 等一下详细的说明一下
堆建立完毕之后, 我们将顶部元素跟最后一个元素进行交换,
控制最后一个范围的位置, 然后向下调整, 循环下去…

堆排序的原理就是上面所说的那样, 实现的逻辑见下
从顶部向下建堆

 //堆排序的从上往下建堆的实现方式
    public void heapSort(int[] nums){
        //首先需要的是建堆
        int len = nums.length;
        for (int i = 0; i < len; i++) {
            heapInsert(nums,i);
        }

        //建堆完毕之后, 进行堆排序的过程
        while(len > 0){
            swap(nums,0,--len);
            heapify(nums,0,len);
        }
    }

上面的从上往下建堆的时间复杂度是O(NLogN)
这个数学证明我们就不说了
大概原理就是 : log1 + log2 + log3 +…+ logN 收敛于N
LogN
但是下面进行交换排序的时间复杂度是一定是O(N*LogN)

堆排序的从底部向上建堆的代码实现

  //从顶部向下建堆的方式
    public void heapSort1(int[] nums) {
        //建堆成功
        for (int pra = (nums.length - 1 - 1) / 2; pra >= 0; pra--) {
            heapify(nums,pra,nums.length);
        }

        //建堆完毕之后, 进行堆排序的过程
        int len = nums.length;
        while (len > 0) {
            swap(nums, 0, --len);
            heapify(nums, 0, len);
        }
    }

为什么说从底部向上建堆的时间复杂度更好
每一层累加起来其实是一个差比数列求和公式
下图左边的是看的节点的规模, 右侧的是向下调整的层数

在这里插入图片描述

本节的代码实现合集


/**
 * 下面我们自己实现一个堆
 */
class Heap {
    int[] elem;
    int usedSize;
    private static final int DEFAULT_CAPACITY = 11;//底层默认是11

    //提供两个构造方法
    public Heap() {
        this(DEFAULT_CAPACITY);
    }

    public Heap(int initCapacity) {
        elem = new int[initCapacity];
    }

    /**
     * 建堆的时候, 我们尝试直接用数组来模拟实现大根堆
     * 之前我们一直不理解, 为什么建堆要从后往前逐一的向下调整,
     * 现在明白了,因为这样才能保证每次调整完毕之后下一次调整的时候只有一个顶部元素是不确定的
     *
     * @param nums
     */
    public Heap(int[] nums) {
        //空间不够就进行扩容(我们现在实现的这个只是简单的版本,可能会因为空间不足的bug)
        this(DEFAULT_CAPACITY);
        if (elem.length <= nums.length) {
            this.elem = Arrays.copyOf(elem, elem.length * 2);
        }

        //进行拷贝元素
        for (int i = 0; i < nums.length; ++i) {
            elem[i] = nums[i];
        }
        usedSize = nums.length;

        //下面就进行建堆的过程
        for (int parent = (usedSize - 1 - 1) / 2; parent >= 0; --parent) {
            heapify(elem, parent, usedSize);
        }
    }

    /**
     * 堆的调正是整个堆操作的最重要的步骤(这里我们默认实现的是大根堆)
     * --> 根据之前我们学习堆结构时期的困惑, 不管是向上调整还是向下调整,都是因为只有某个位置的元素破坏了堆的结构, 切记是一个! ! !
     *
     * @param nums : 传入的需要调整的数组
     * @param i    : 待调整元素的下标
     *             向上调整不需要设置终点位置, 因为就是根节点自己 当 i == 0是时, (i - 1) / 2 == 0,自动跳出循环
     */
    private void heapInsert(int[] nums, int i) {
        while (nums[i] > nums[(i - 1) / 2]) {
            swap(nums, i, (i - 1) / 2);
            i = (i - 1) / 2;
        }
    }


    /**
     * 这里跟上面的向上调整的类似的,我们都是只有其中的一个元素破坏了堆的结构(默认实现大根堆)
     * 之前我们没有理解向上调整跟向下调整的原因就是,我们忽略了只有一个元素破坏了堆结构这一条件
     *
     * @param nums : 待调整的数组
     * @param i    : 待调整的元素的下标
     * @param end  : 结束位置
     */
    private void heapify(int[] nums, int i, int end) {
        int l = 2 * i + 1; //先把孩子下标打到左孩子的位置;
        while (l < end) {
            //下面这句话的代码逻辑是 --> 找到最大孩子的孩子下标
            int best = l + 1 < end && nums[l + 1] > nums[l] ? l + 1 : l;
            //下面这句话的代码逻辑是 --> 比较最大的孩子跟父节点
            best = nums[i] < nums[best] ? best : i;
            if (best == i) {
                break;
            }
            swap(nums, best, i);
            i = best;
            l = 2 * i + 1;
        }
    }

    private void swap(int[] nums, int i, int j) {
        int temp = nums[i];
        nums[i] = nums[j];
        nums[j] = temp;
    }


    /**
     * 下面是简单的删除操作
     */
    public void poll() {
        swap(elem, 0, usedSize - 1);
        usedSize--;
        heapify(elem, 0, usedSize);
    }

    /**
     * 一些其他的操作
     */
    public int peek() {
        return elem[0];
    }

    //展示的操作
    public void show() {
        for (int i = 0; i < usedSize; i++) {
            System.out.print(elem[i] + " ");
        }
        System.out.println();
    }

    //堆排序的从上往下建堆的实现方式
    public void heapSort(int[] nums) {
        //首先需要的是建堆
        int len = nums.length;
        for (int i = 0; i < len; i++) {
            heapInsert(nums, i);
        }

        //建堆完毕之后, 进行堆排序的过程
        while (len > 0) {
            swap(nums, 0, --len);
            heapify(nums, 0, len);
        }
    }

    //从顶部向下建堆的方式
    public void heapSort1(int[] nums) {
        //建堆成功
        for (int pra = (nums.length - 1 - 1) / 2; pra >= 0; pra--) {
            heapify(nums,pra,nums.length);
        }

        //建堆完毕之后, 进行堆排序的过程
        int len = nums.length;
        while (len > 0) {
            swap(nums, 0, --len);
            heapify(nums, 0, len);
        }
    }

    public static void main(String[] args) {
        int[] nums = new int[]{5, 8, 6, 7, 3, 1, 9, 2, 7, 1, 4, 6};
        Heap heap = new Heap(nums);
        heap.heapSort1(nums);
        System.out.println(Arrays.toString(nums));
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【计算机毕设】基于SpringBoot的教学资源库设计与实现 - 源码免费(私信领取)

免费领取源码 &#xff5c; 项目完整可运行 &#xff5c; v&#xff1a;chengn7890 诚招源码校园代理&#xff01; 1. 研究目的 本项目旨在设计并实现一个基于SpringBoot的教学资源库系统&#xff0c;以便教师和学生能够方便地存储、分享和查找各种教学资源。具体目标包括&…

分治策略的实现

目录 前言 分治策略的应用 最大子数组问题 矩阵乘法问题 求解递归式的三种方法 代入法求递归式 用递归树求递归式 主方法求递归式 前言 分治三个步骤&#xff1a; 分解&#xff1a;分解原问题为子问题&#xff0c;这些子问题为原问题的较小规模的问题。 解决&#xf…

Redis——基本命令

概念&#xff1a; Redis(REmote Dlctionary Server) 是用 C语言开发的一个开源的高性能键值对(key-value) 数据库 特征&#xff1a; 1. 数据间没有必然的关联关系 2. 内部采用单线程机制进行工作 3. 高性能 4. 多数据类型支持 字符串类型 string 列表类型 …

新 Google 邮箱注册的美区Appleid 账户被停用如何解冻?

什么条件触发美区账号被停用&#xff1f; 如何触发的被停用&#xff0c;我猜是因为新账户没有进行安全认证&#xff0c;在新机器手机上登陆&#xff0c;下载app导致的。 如何解冻美区 Appleid 账户&#xff1f; 打苹果服务支持电话&#xff1a;4006668800 苹果员工会非常耐心…

ios 新安装app收不到fcm推送

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

Charles的安装和web端抓包配置

1.Charles的安装 通过官网下载&#xff1a;https://www.charlesproxy.com/download/&#xff0c;我之前下载的是4.6.2版本&#xff0c;下载成功后点击安装包&#xff0c;点击下一步下一步即可安装成功。 ​​ ​ 安装成功后打开charles页面如下所示。 ​ 2.乱码问题解决 打开…

【Docker学习】docker pull详细说明

docker pull是我们经常用到的一个命令。我们使用一些官方镜像&#xff0c;如MySql、Nginx等都需要用docker pull下载。不过不用的话&#xff0c;也可以。比如使用docker run&#xff0c;要是找不到镜像&#xff0c;会自动下载。 命令&#xff1a; docker image pull 描述&am…

插入排序以及希尔排序; 先学会插入,希尔会更简单喔

1.前言 首先肯定是要学会插入排序再学习希尔排序会更简单&#xff0c;因为代码部分有很多相似之处&#xff1b;如果你觉得你很强&#xff0c;可以直接看希尔排序的讲解。哈哈哈&#xff01;&#xff0c;每天进步一点点&#xff0c;和昨天的自己比 2.插入排序 让我们先来看看…

【Hive SQL 每日一题】统计每月用户购买商品的种类分布

文章目录 测试数据需求说明需求实现 测试数据 -- 创建 orders 表 DROP TABLE IF EXISTS orders; CREATE TABLE orders (order_id INT,user_id INT,product_id INT,order_date STRING );-- 插入 orders 数据 INSERT INTO orders VALUES (101, 1, 1001, 2023-01-01), (102, 1, 1…

pycharm简易使用码云gitee

文章目录 参考文献官网地址安装插件第一个选项报错了不可&#xff0c;第二个选项&#xff0c;可以了新库上传到主分支&#xff0c;push改进实验新建分支&#xff0c;上传为新分支&#xff1a;做另一种改进&#xff0c;选择回退主分支&#xff0c;另建一个分支 使用对于一个新项…

SQL158 每类视频近一个月的转发量/率

描述 用户-视频互动表tb_user_video_log iduidvideo_idstart_timeend_timeif_followif_likeif_retweetcomment_id110120012021-10-01 10:00:002021-10-01 10:00:20011NULL210220012021-10-01 10:00:002021-10-01 10:00:15001NULL310320012021-10-01 11:00:502021-10-01 11:01…

Python 基于机器学习模型的车牌检测和识别系统 有GUI界面 【含Python源码 MX_004期】

一、系统介绍 车牌的检测和识别技术在现代社会中的应用场景可谓十分广泛&#xff0c;不仅涉及交通管理领域&#xff0c;还延伸至社区安保等多个方面。例如&#xff0c;在交通违章管理中&#xff0c;通过车牌追踪可以有效追踪违章车辆&#xff0c;维护交通秩序&#xff1b;在小区…

【UML用户指南】-05-对基本结构建模-类

在UML中&#xff0c;所有的事物都被建模为类。类是对词汇表中一些事物的抽象。类不是个体对象&#xff0c;而是描述一些对象的一个完整集合。 强调抽象的最重要的部分&#xff1a;名称、属性和操作 类 &#xff08;class&#xff09;是对一组具有相同属性、操作、关系和语义的对…

JVM垃圾收集器和内存分配策略

概述 Java内存运行时数据区的程序计数器、虚拟机栈、本地方法栈3个区域会随着线程而产生&#xff0c;随线程而消失。这几个区域分配多少内存时在类结构确定下来即已知的&#xff0c;在这几个区域内就不需要过多考虑如何回收内存的问题&#xff0c;当方法结束或者线程结束时&am…

第三届大湾区算力大会丨暴雨开启数字未来新篇

5月30-31日&#xff0c;韶关市迎来主题为“算启新篇智创未来”的第三届粤港澳大湾区(广东)算力产业大会暨第二届中国算力网大会&#xff0c;活动由广东省人民政府主办&#xff0c;广东省政数局、韶关市人民政府共同承办。暴雨信息作为算力产业发展的重要构建者受邀赴会&#xf…

【C++进阶】深入STL之string:模拟实现走进C++字符串的世界

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ ⏩收录专栏⏪&#xff1a;C “ 登神长阶 ” &#x1f921;往期回顾&#x1f921;&#xff1a;C模板入门 &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀STL之string &#x1f4d2;1. string…

力扣575. 分糖果

题目&#xff1a; Alice 有 n 枚糖&#xff0c;其中第 i 枚糖的类型为 candyType[i] 。Alice 注意到她的体重正在增长&#xff0c;所以前去拜访了一位医生。 医生建议 Alice 要少摄入糖分&#xff0c;只吃掉她所有糖的 n / 2 即可&#xff08;n 是一个偶数&#xff09;。Alic…

Java中连接Mongodb进行操作

文章目录 1.引入Java驱动依赖2.快速开始2.1 先在monsh连接建立collection2.2 java中快速开始2.3 Insert a Document2.4 Update a Document2.5 Find a Document2.6 Delete a Document 1.引入Java驱动依赖 注意&#xff1a;启动服务的时候需要加ip绑定 需要引入依赖 <dependen…

Java的数据库编程-----JDBC

目录 一.JDBC概念&使用条件&#xff1a; 二.mysql-connector驱动包的下载与导入&#xff1a; 三.JDBC编程&#xff1a; 使用JDBC编程的主要五个步骤&#xff1a; 完整流程1&#xff08;更新update&#xff09;&#xff1a; 完整流程2(查询query)&#xff1a; 一.JDB…

FS212E 系列PD协议

PD快充协议芯片FS212EL、FS212EH可以智能的识别插入的手机类型&#xff0c;选择最为合适的协议应对手机快充需要。兼容多类USB Type-C协议&#xff0c;包括TypeC协议、TypeC PD2.0、TypeC PD3.0、TypeC PD3.2等协议。集成OPTO输出&#xff0c;通过电阻直驱反馈光耦。FS212E 的调…