彻底搞定堆排序:二叉堆
作者:程序dunk 发布时间:2022-12-06 23:24:15
标签:堆排序,二叉堆
二叉堆
什么是二叉堆
二叉堆本质上是一种完全二叉树,它分为两个类型
最大堆:最大堆的任何一个父节点的值,都大于等于它的左、右孩子节点的值(堆顶就是整个堆的最大元素)
最小堆:最小堆的任何一个父节点的值,都小于等于它的左、右孩子节点的值(堆顶就是整个堆的最小元素)
二叉堆的根节点叫做堆顶
二叉堆的基本操作
插入节点
删除节点
构建二叉堆
这几种操作都基于堆的自我调整,所谓堆自我调整,就是把一个不符合堆的完全二叉树,调整成一个堆,下面以最小堆为例。
插入
插入节点0的过程
删除
删除节点的过程和插入的过程刚好相反,所删除的是处于堆顶的节点。例如删除1
为了维持完全二叉树的结构,把堆的最后一个元素临时补充到堆顶
删除原来10的位置
对堆顶的节点10执行下沉操作
构建
构建二叉堆,也就是把一个无序的完全二叉树调整为二叉堆,本质就是让所有的非叶子节点一次下沉
二叉堆代码实现
二查堆虽然是一颗完全二叉树,但它的存储方式并不是链式的,而是顺序存储,换句话说,二叉堆的所有节点都存储在数组中
当父节点为parent时,左孩子为2 * parent + 1;右孩子为2 * parent + 2
/**
* @author :zsy
* @date :Created 2021/5/17 9:41
* @description:二叉堆
*/
public class HeapTest {
public static void main(String[] args) {
int[] arr = {1, 3, 2, 6, 5, 7, 8, 9, 10, 0};
Heap heap = new Heap(arr);
heap.upAdjust(arr);
System.out.println(Arrays.toString(arr));
arr = new int[]{7, 1, 3, 10, 5, 2, 8, 9, 6};
heap = new Heap(arr);
heap.buildHead();
System.out.println(Arrays.toString(arr));
}
}
class Heap {
private int[] arr;
public Heap(int[] arr) {
this.arr = arr;
}
public void buildHead() {
//从最后一个非叶子节点开始,依次下沉
for (int i = (arr.length - 2) / 2; i >= 0; i--) {
downAdjust(arr, i, arr.length);
}
}
private void downAdjust(int[] arr, int parentIndex, int length) {
int temp = arr[parentIndex];
int childrenIndex = parentIndex * 2 + 1;
while (childrenIndex < length) {
//如果有右孩子,并且右孩子小于左孩子,那么定位到右孩子
if (childrenIndex + 1 < length && arr[childrenIndex + 1] < arr[childrenIndex]) {
childrenIndex++;
}
//如果父节点小于较小孩子节点的值,直接跳出
if (temp <= arr[childrenIndex]) break;
//无需交换,单向赋值
arr[parentIndex] = arr[childrenIndex];
parentIndex = childrenIndex;
childrenIndex = 2 * childrenIndex + 1;
}
arr[parentIndex] = temp;
}
public void upAdjust(int[] arr) {
int childrenIndex = arr.length - 1;
int parentIndex = (childrenIndex - 1) / 2;
int temp = arr[childrenIndex];
while (childrenIndex > 0 && temp < arr[parentIndex]) {
//单向赋值
arr[childrenIndex] = arr[parentIndex];
childrenIndex = parentIndex;
parentIndex = (parentIndex - 1) / 2;
}
arr[childrenIndex] = temp;
}
}
结果:
[0, 1, 2, 6, 3, 7, 8, 9, 10, 5]
[1, 5, 2, 6, 7, 3, 8, 9, 10]
来源:https://blog.csdn.net/qq_45796208/article/details/117094130


猜你喜欢
- 本文实例为大家分享了WPF实现文字粒子闪烁动画的具体代码,供大家参考,具体内容如下实现效果如下:思路:首先根据显示文本创建文本路径Geome
- HttpServletResponse.sendRedirect与RequestDispatcher.forward方法都可以实
- 本文实例为大家分享了opencv实现轮廓高斯滤波平滑的具体代码,供大家参考,具体内容如下一个小测试的题目:在图像上点选,找到与点选处相邻的颜
- 单例:Singleton,是指仅仅被实例化一次的类。饿汉单例设计模式一、饿汉设计模式public class SingletonHungry
- 1.aar包是android studio下打包android工程中src、res、lib后生成的aar文件,aar包导入其他android
- 博主第一次安装Android Studio 3.6版本的时候就找不到R.java文件,于是在网上找个各种方法,但是都没能解决问题。注意:本博
- 概念final 具有“不可改变的”的含义,可以修饰 非抽象类、非抽象成员方法和变量。用 final 修饰的类不能被继承,没有子类。用 fin
- 题目一 解法/** * Definition for a binary tree node. * public class Tre
- 纯Java代码模拟Hibernate一级缓存原理,简单易懂。import java.util.ArrayList;import java.u
- 我们肯定遇到过打开别人的项目时一直处于Building‘XXX'Gradle project info的情况。本文通过两种方法带领大
- #region 提示信息#endregion作用:折叠并隐藏代码 ,别且折叠以后能够显示白字“提示信息”如下图就是使用了#region和#e
- 推荐第三种方式,简单快捷不卡。第一种:jjdxm_updateGitHub地址:jjdxmashl/jjdxm_update效果图:点击立即
- 必须先要了解的1。c/c++是程序员自己管理内存,Java内存是由GC自动回收的。我虽然不是很熟悉C++,不过这个应该没有犯常识性错误吧。2
- 一、使用spring initializr创建java工程 1、启动IDEA,新建java工程,使用向导创建一个springboo
- 1. MyBatis 中 #{}和 ${}的区别是什么?#{}是预编译处理,${}是字符替换。 在使用 #{}时,MyBatis 会将 SQ
- 这篇文章主要介绍了java操作elasticsearch的案例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价
- 第一步新建txt文件,写入内容我是放在D盘下的,数据以逗号隔开的,是英文逗号第二步读取数据在需要读取数据的页面,添加代码,就可以了 priv
- java随机验证码生成实现实例代码摘要: 在项目中有很多情况下都需要使用到随机验证码,这里提供一个java的随机验证码生成方案,可以指定难度
- 最近一直在对接接口,上游返回的都是 JSON 数据,我们需要将这些数据进行保存,我们可以解析成 Map 通过 key 的方式进行获取,然后
- package com.example.myapi.email;import java.util.ArrayList;import java