java 堆 排序学习
/**
* <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 堆 排序学习的更多相关文章
- 从几个sample来学习JAVA堆、方法区、JAVA栈和本地方法栈
最近在看<深入理解Java虚拟机>,书中给了几个例子,比较好的说明了几种OOM(OutOfMemory)产生的过程,大部分的程序员在写程序时不会太关注Java运行时数据区域的结构: 感觉有 ...
- 从sample来学习Java堆(转)
1)Java堆 所有对象的实例分配都在Java堆上分配内存,堆大小由-Xmx和-Xms来调节,sample如下所示: public class HeapOOM { static class OOMOb ...
- JVM学习--(八)java堆分析
上一节介绍了针对JVM的监控工具,包括JPS可以查看当前所有的java进程,jstack查看线程栈可以帮助你分析是否有死锁等情况,jmap可以导出java堆文件在MAT工具上进行分析等等.这些工具都非 ...
- 《深入理解java虚拟机》学习笔记之编译优化技术
郑重声明:本片博客是学习<深入理解Java虚拟机>一书所记录的笔记,内容基本为书中知识. Java程序员有一个共识,以编译方式执行本地代码比解释方式更快,之所以有这样的共识,除去虚拟机解释 ...
- 《深入理解Java虚拟机》学习笔记
<深入理解Java虚拟机>学习笔记 一.走近Java JDK(Java Development Kit):包含Java程序设计语言,Java虚拟机,JavaAPI,是用于支持 Java 程 ...
- Java虚拟机(JVM) - 学习总结(全)
深入理解java虚拟机---学习总结: 1.Java内存区域 1.1 java运行时数据区 Java 虚拟机所管理的内存如下图所示,基于JDK1.6. 基于jdk1.8画的JVM的内存模型 (1) 程 ...
- 《深入理解 java虚拟机》学习笔记
java内存区域详解 以下内容参考自<深入理解 java虚拟机 JVM高级特性与最佳实践>,其中图片大多取自网络与本书,以供学习和参考.
- Java程序员学习之路
1. Java语言基础 谈到Java语 言基础学习的书籍,大家肯定会推荐Bruce Eckel的<Thinking in Java>.它是一本写的相当深刻的技术书籍,Java语言基础部分基 ...
- Java虚拟机JVM学习07 类的卸载机制
Java虚拟机JVM学习07 类的卸载机制 类的生命周期 当Sample类被加载.连接和初始化后,它的生命周期就开始了. 当代表Sample类的Class对象不再被引用,即不可触及时,Class对象就 ...
随机推荐
- Shell的语法
Shell的语法: 变量:字符串.数字.环境和参数: 条件:shell中的布尔值: 程序控制:if.elif.for.while.until.case: 命令列表: 函数: Shell内置命令: 获取 ...
- 差分形式的牛顿插值法(c++)
本程序对cosx函数进行插值,取步长为0.1,因此x的值为0.00,0.10,0.20,0.30,对应的y值为cos(0.00),cos(0.10),cos(0.20),cos(0.30),其实本程序 ...
- ICEM-空心圆柱体
原视频下载地址:https://pan.baidu.com/s/1boG49MB 密码: 4iq6
- 城东C位之路!探秘三线楼市板块崛起3大核心基因
等着咧!伍家篇什么时候出?这就出. 城东C位之路!- 诸葛磊 好几个粉丝已经在催了,诸葛磊决定改变下写作策略,伍家岗篇分版块用小篇幅来写,这样文章不至于太长,否则又是一篇洋洋洒洒上万字的文章,粉丝看着 ...
- Learning Context Graph for Person Search
Learning Context Graph for Person Search 2019-06-24 09:14:03 Paper:http://openaccess.thecvf.com/cont ...
- locust参数化(数据库取值)
locust参数化(数据库取值) 基于上一篇参数化的梳理,本篇用另一种方法从数据库中取出这100个用户来登录 思路:在 TaskSet 中的 on_start 方法表示执行任务前的操作,可以将数据库取 ...
- java里的数组和list分别在什么情况下使用?
数组长度固定,List未限定长度,且支持的功能更多,最常用的ArrayList底层实际上也是使用数组实现. 不需要复杂功能和确定长度的情况下,使用数组效率更高,通常情况建议使用List.
- (十三)GBDT模型用于评分卡模型python实现
python信用评分卡建模(附代码,博主录制) https://study.163.com/course/introduction.htm?courseId=1005214003&utm_ca ...
- js知识体系
- hibernate的load和get有什么作用
① load方法认为该数据在数据库中一定存在,可以放心的使用代理来延迟加载,如果在使用过程中发现了问题,只能抛异常(ObjectNotFoundException)load方法加载实体对象的时候,根据 ...