/**
* <html>
* <body>
* <P> Copyright 1994 JsonInternational</p>
* <p> All rights reserved. - https://github.com/Jasonandy/Java-Core-Advanced </p>
* <p> Created by Jason</p>
* </body>
* </html>
*/
package cn.ucaner.datastructure.heap; /**
* @Package:cn.ucaner.datastructure.heap
* @ClassName:MinHeap
* @Description: <p> 最小堆 :完全二叉树,能方便地从中取出最小/大元素 </p>
* 堆的构建
* 堆的打印(前序遍历的应用)
* 堆的插入(插入到堆尾,再自下向上调整为最小堆)
* 堆的删除(删除堆顶元素并用堆尾元素添补,再自上向下调整为最小堆)
* 堆排序(时间复杂度:O(nlgn),空间复杂度O(1),不稳定):升序排序一般用最大堆
* @Author: -
* @CreatTime:2018年6月8日 上午10:48:46
* @Modify By:
* @ModifyTime: 2018年6月8日
* @Modify marker:
* @version V1.0
*/
public class MinHeap { /**
* 将所有元素以完全二叉树的形式存入数组
*/
private int[] heap; /**
* 堆中元素的个数
*/
private int size; /**
* MinHeap. 构造函数 - 构建一个大小为size的最小堆
* @param maxSize
*/
public MinHeap(int maxSize) {
heap = new int[maxSize];
} /**
* MinHeap. 构造函数
* @param arr 基于数组构造最小堆
* @param maxSize
*/
public MinHeap(int[] arr, int maxSize) {
heap = new int[maxSize > arr.length ? maxSize : arr.length];
System.arraycopy(arr, 0, heap, 0, arr.length);
size = arr.length; int pos = (size - 2) / 2; // 最初调整位置:最后的分支节点(最后叶节点的父亲)
while (pos >= 0) { //依次调整每个分支节点
shiftDown(pos, size - 1);
pos--;
}
} /**
* @Description: 自上向下调整为最小堆(从不是最小堆调整为最小堆),调整的前提是其左子树与右子树均为最小堆
* @param start
* @param end void
* @Autor: jason - jasonandy@hotmail.com
*/
private void shiftDown(int start, int end) {
int i = start; // 起始调整位置,分支节点
int j = 2 * start + 1; // 该分支节点的子节点
int temp = heap[i];
while (j <= end) { // 迭代条件:子节点不能超出end(范围)
if (j < end) {
j = heap[j] > heap[j + 1] ? j + 1 : j; // 选择两孩子中较小的那个
}
if (temp < heap[j]) { // 较小的孩子大于父亲,不做任何处理
break;
} else { // 否则,替换父节点的值
heap[i] = heap[j];
i = j;
j = 2 * j + 1;
}
}
heap[i] = temp; // 一步到位
} /**
* @Description: 自下向上调整为最小堆(原来已是最小堆,添加元素后,确保其还是最小堆)
* @Autor:jason - jasonandy@hotmail.com
*/
private void shiftUp(int start) {
int j = start;
int i = (j - 1) / 2; // 起始调整位置,分支节点
int temp = heap[j];
while (j > 0) { // 迭代条件:子节点必须不为根
if (temp >= heap[i]) { //原已是最小堆,所以只需比较这个子女与父亲的关系即可
break;
} else {
heap[j] = heap[i];
j = i;
i = (j - 1) / 2;
}
}
heap[j] = temp; // 一步到位
} /**
* @Description: 向最小堆插入元素(总是插入到最小堆的最后)
* @param data
* @Autor: jason - jasonandy@hotmail.com
*/
public void insert(int data){
if (size < heap.length) {
heap[size++] = data; // 插入堆尾
shiftUp(size-1); // 自下而上调整
}
} /**
* @Description:删除堆顶元素,以堆的最后一个元素填充
* @Autor: jason - jasonandy@hotmail.com
*/
public void remove() {
if (size > 0) {
heap[0] = heap[size-1]; // 删除堆顶元素,并将堆尾元素回填到堆顶
size --; // 堆大小减一
shiftDown(0, size-1); // 自上向下调整为最小堆
}
} /**
* @Description: 堆排序:每次将最小元素交换到最后
* @Autor: jason - jasonandy@hotmail.com
*/
public void sort(){
for (int i = size - 1; i >= 0; i--) {
int temp = heap[0];
heap[0] = heap[i];
heap[i] = temp; shiftDown(0, i-1);
} for (int i = size-1; i >= 0; i--) {
System.out.print(heap[i] + " ");
}
} /**
* @Description: 打印根为 i 的最小堆
* @param i
* @Autor: Jason - jasonandy@hotmail.com
*/
public void printMinHeap(int i) {
if (size > i) {
System.out.print(heap[i]);
if (2 * i + 1 < size || 2 * i + 2 < size) {
System.out.print("(");
printMinHeap(2 * i + 1);
System.out.print(",");
printMinHeap(2 * i + 2);
System.out.print(")");
}
}
}
}

java 堆 排序学习的更多相关文章

  1. 从几个sample来学习JAVA堆、方法区、JAVA栈和本地方法栈

    最近在看<深入理解Java虚拟机>,书中给了几个例子,比较好的说明了几种OOM(OutOfMemory)产生的过程,大部分的程序员在写程序时不会太关注Java运行时数据区域的结构: 感觉有 ...

  2. 从sample来学习Java堆(转)

    1)Java堆 所有对象的实例分配都在Java堆上分配内存,堆大小由-Xmx和-Xms来调节,sample如下所示: public class HeapOOM { static class OOMOb ...

  3. JVM学习--(八)java堆分析

    上一节介绍了针对JVM的监控工具,包括JPS可以查看当前所有的java进程,jstack查看线程栈可以帮助你分析是否有死锁等情况,jmap可以导出java堆文件在MAT工具上进行分析等等.这些工具都非 ...

  4. 《深入理解java虚拟机》学习笔记之编译优化技术

    郑重声明:本片博客是学习<深入理解Java虚拟机>一书所记录的笔记,内容基本为书中知识. Java程序员有一个共识,以编译方式执行本地代码比解释方式更快,之所以有这样的共识,除去虚拟机解释 ...

  5. 《深入理解Java虚拟机》学习笔记

    <深入理解Java虚拟机>学习笔记 一.走近Java JDK(Java Development Kit):包含Java程序设计语言,Java虚拟机,JavaAPI,是用于支持 Java 程 ...

  6. Java虚拟机(JVM) - 学习总结(全)

    深入理解java虚拟机---学习总结: 1.Java内存区域 1.1 java运行时数据区 Java 虚拟机所管理的内存如下图所示,基于JDK1.6. 基于jdk1.8画的JVM的内存模型 (1) 程 ...

  7. 《深入理解 java虚拟机》学习笔记

    java内存区域详解 以下内容参考自<深入理解 java虚拟机 JVM高级特性与最佳实践>,其中图片大多取自网络与本书,以供学习和参考.

  8. Java程序员学习之路

    1. Java语言基础 谈到Java语 言基础学习的书籍,大家肯定会推荐Bruce Eckel的<Thinking in Java>.它是一本写的相当深刻的技术书籍,Java语言基础部分基 ...

  9. Java虚拟机JVM学习07 类的卸载机制

    Java虚拟机JVM学习07 类的卸载机制 类的生命周期 当Sample类被加载.连接和初始化后,它的生命周期就开始了. 当代表Sample类的Class对象不再被引用,即不可触及时,Class对象就 ...

随机推荐

  1. VS2017 Asp.Net调式闪退处理

  2. .getCellType()的几种类型值

    CellType 类型 值   CELL_TYPE_NUMERIC 数值型 0 CELL_TYPE_STRING 字符串型 1  CELL_TYPE_FORMULA 公式型       2   CEL ...

  3. ICEM-空心圆柱体

    原视频下载地址:https://pan.baidu.com/s/1boG49MB 密码: 4iq6

  4. ORA-55624: 此时无法为闪回归档启用表

    我们在某应用中使用了FDA特性,但是某些表在解除归档后重新启用时报"ORA-55624: 此时无法为闪回归档启用表",经查询google和MOS相关信息,原因就是太频繁.解决方法: ...

  5. 纯Python模式

    http://crcmod.sourceforge.net/intro.html https://help.aliyun.com/document_detail/85288.html OSS的CRC数 ...

  6. Ingress-nginx 部署使用

    Ingress-nginx 部署使用   一.Ingress 简介 在Kubernetes中,服务和Pod的IP地址仅可以在集群网络内部使用,对于集群外的应用是不可见的.为了使外部的应用能够访问集群内 ...

  7. jeff dean的主页

    https://ai.google/research/people/jeff/ 上面有他的很多论文和ppt 很不错 我于1999年中期加入Google,目前是研究小组的Google高级研究员,负责Go ...

  8. ajax发送json数据时为什么需要设置contentType: "application/json”

    1. ajax发送json数据时设置contentType: "application/json”和不设置时到底有什么区别?contentType: "application/js ...

  9. JS高级:面向对象解析

    1 实例属性/方法 都是绑定在使用构造函数创建出来的对象p上; 最终使用的时候也是使用对象p来进行访问; function Person(name, age, doFunc) { this.name ...

  10. 【转】HTML meta标签总结与属性使用介绍

    HTML meta标签总结与属性使用介绍 转载处写的已经超级好了,强烈推荐. 转自:https://segmentfault.com/a/1190000004279791 本人就不再赘述.拿来主义!供 ...