造轮子-Java泛型堆排
个人博客地址:http://kyle.org.cn/2018/03/13/heapsort/
Java实现泛型堆排算法,用于N个对象中选择最大或者最小的前M个,其中M<=N
类似于Mysql中order by + limit的功能,如果有类似场景的需求,可以直接拷贝到项目中使用
Github源码地址:https://github.com/Kyle-Wilson1/Algorithm_Java/tree/master/heapsort
工程目录结构
- BootStrap:启动类,测试入口
- Node:排序的对象
- AbstractHeapObject:对象需要实现的接口
- HeapDirectionEnum:堆方向枚举
- HeapSort:堆排算法实现类
- HomeController:web api测试
启动类
- 作为程序的入口,写了一些测试数据
public class BootStrap {
public static void main(String[] args) {
new BootStrap().handle();
}
private void handle() {
HeapSort<Node> heapSort = new HeapSort<>();
List<Node> arr = new ArrayList<>(Arrays.asList(
new Node(1L, 1L),
new Node(2L, 2L),
new Node(1L, 1L),
new Node(2L, 2L),
new Node(3L, 3L),
new Node(4L, 4L)));
heapSort.setDirection(HeapDirectionEnum.MAX_ROOT);
heapSort.setHeapCapability(5);
heapSort.buildHeap(arr);
heapSort.getHeap().forEach(System.out::println);
List<Node> arr1 = new ArrayList<>(Arrays.asList(
new Node(1L, 1L),
new Node(2L, 2L),
new Node(3L, 3L),
new Node(4L, 4L),
new Node(5L, 5L)));
heapSort.buildHeap(arr1);
System.out.println("insert arr1:");
heapSort.getHeap().forEach(System.out::println);
System.out.println("pop: " + heapSort.heapPop());
}
}
测试结果如下:
Node{key=3, value=3}
Node{key=2, value=2}
Node{key=1, value=1}
Node{key=1, value=1}
Node{key=2, value=2}
insert arr1:
Node{key=2, value=2}
Node{key=2, value=2}
Node{key=1, value=1}
Node{key=1, value=1}
Node{key=1, value=1}
pop: Node{key=2, value=2}
定义对象接口
- 对于要排序的对象,必须先实现该接口
- getKey()返回某一个对象要用于排序的字段
public abstract class AbstractHeapObject {
public abstract Long getKey();
}
排序对象
- 待排序的类实现,需要实现AbstractHeapObject接口
public class Node extends AbstractHeapObject {
private Long key;
private Long value;
public Node(Long key, Long value) {
this.key = key;
this.value = value;
}
@Override
public Long getKey() {
return key;
}
public Long getValue() {
return value;
}
@Override
public String toString() {
return "Node{" +
"key=" + key +
", value=" + value +
'}';
}
}
定义堆排方向枚举
- 大根堆,适用于N个数中求最小的前M个数
- 小根堆,适用于N个数中求最大的前M个数
public enum HeapDirectionEnum {
//大根堆,适用于N个数中求最小的前M个数
MAX_ROOT(0, "MAX_ROOT", "大根堆"),
//小根堆,适用于N个数中求最大的前M个数
MIN_ROOT(1, "MIN_ROOT", "小根堆");
private Integer id;
private String name;
private String remark;
HeapDirectionEnum(Integer id, String name, String remark) {
this.id = id;
this.name = name;
this.remark = remark;
}
public Integer getId() {
return id;
}
public String getName() {
return name;
}
public String getRemark() {
return remark;
}
}
堆排核心算法
- 堆的向上调整,向下调整,弹出堆等实现
public class HeapSort<T extends AbstractHeapObject> {
private List<T> sourceData;
private HeapDirectionEnum direction;
private List<T> heap;
private int heapCapability;
public void setDirection(HeapDirectionEnum direction) {
this.direction = direction;
}
public void setHeapCapability(int heapCapability) {
this.heapCapability = heapCapability;
}
public List<T> getHeap() {
return heap;
}
synchronized public void buildHeap(List<T> sourceData) {
this.sourceData = sourceData;
//初始化堆容量
if (heap == null) {
heap = new ArrayList<>(heapCapability);
}
sourceData.forEach(x -> {
//如果堆中元素还没有达到最大值,则插入到堆尾并向上调整,否则替换根元素并向下进行调整
if (heap.size() < heapCapability) {
heapUp(x);
} else {
if (direction == HeapDirectionEnum.MAX_ROOT && x.getKey() < heap.get(0).getKey()
|| direction == HeapDirectionEnum.MIN_ROOT && x.getKey() > heap.get(0).getKey()) {
heap.set(0, x);
heapDown(0);
}
}
});
}
/**
* @param current the value to be added at the tail
*/
private void heapUp(T current) {
int i, j;
heap.add(current);
i = heap.size() - 1;
j = (i - 1) / 2; //j指向i的父结点
while (i > 0) {
if (direction == HeapDirectionEnum.MAX_ROOT && heap.get(j).getKey() >= current.getKey()
|| direction == HeapDirectionEnum.MIN_ROOT && heap.get(j).getKey() <= current.getKey()) {
break;
}
heap.set(i, heap.get(j));
i = j;
j = (i - 1) / 2;
}
heap.set(i, current);
}
/**
* @param top the location where the value will be adjusted down
*/
private void heapDown(int top) {
int j = 2 * top + 1;
T x = heap.get(top);
int heapSize = heap.size() - 1;
while (j <= heapSize) {
if (j + 1 <= heapSize && (
direction == HeapDirectionEnum.MAX_ROOT && heap.get(j + 1).getKey() > heap.get(j).getKey()
|| direction == HeapDirectionEnum.MIN_ROOT && heap.get(j + 1).getKey() < heap.get(j).getKey()))
j++;
if (direction == HeapDirectionEnum.MAX_ROOT && heap.get(j).getKey() <= x.getKey()
|| direction == HeapDirectionEnum.MIN_ROOT && heap.get(j).getKey() >= x.getKey()) {
break;
}
heap.set(top, heap.get(j));
top = j;
j = 2 * top + 1;
}
heap.set(top, x);
}
public T heapPop() {
T ret = heap.get(0);
heap.set(0, heap.get(heap.size() - 1));
heap.remove(heap.size() - 1);
heapDown(0);
return ret;
}
}
造轮子-Java泛型堆排的更多相关文章
- 别在重复造轮子了,几个值得应用到项目中的 Java 开源库送给你
我是风筝,公众号「古时的风筝」.文章会收录在 JavaNewBee 中,更有 Java 后端知识图谱,从小白到大牛要走的路都在里面.公众号回复『666』获取高清大图. 风筝我作为一个野路子开发者,直到 ...
- java 堆排,优先级队列,归并排序
堆排 堆排是基于二叉树而得来的 例如:对一个数组 可以转为二叉树: 二叉树特性父节点为 i , 左叶子节点为2i+1:右叶子节点为2i+2; 步骤分解: 1. 先从第一个非叶子节点(即下 ...
- 避免重复造轮子的UI自动化测试框架开发
一懒起来就好久没更新文章了,其实懒也还是因为忙,今年上半年的加班赶上了去年一年的加班,加班不息啊,好了吐槽完就写写一直打算继续的自动化开发 目前各种UI测试框架层出不穷,但是万变不离其宗,驱动PC浏览 ...
- Java泛型总结
1. 什么是泛型?泛型(Generic type 或者 generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类.可以把类型参数看作是使用参数化类型时指定的类型的 ...
- Java泛型学习笔记--Java泛型和C#泛型比较学习(一)
总结Java的泛型前,先简单的介绍下C#的泛型,通过对比,比较学习Java泛型的目的和设计意图.C#泛型是C#语言2.0和通用语言运行时(CLR)同时支持的一个特性(这一点是导致C#泛型和Java泛型 ...
- 自己造轮子系列之OOM框架AutoMapper
[前言] OOM框架想必大家在Web开发中是使用频率非常之高的,如果还不甚了解OOM框架,那么我们对OOM框架稍作讲解. OOM顾名思义,Object-Object-Mapping实体间相互转换.常见 ...
- 除非你是BAT,前端开发中最好少造轮子
站在前人的肩膀上 HTML.CSS.JavaScript是前端的根基,这是无可否认的事实.正如一辆车当然都是由一堆钢板和螺钉组成的,但是现在还有人拎着个锤子敲敲打打的造车吗?李书福说过,“汽车不过是四 ...
- JAVA 泛型意淫之旅(二)
编译器如何处理泛型 泛型类编译后长什么样? 接上文,JAVA 泛型意淫之旅1,成功迎娶白富美后,终于迎来了最振奋人心的一刻:造娃!造男娃还是造女娃?对于我们程序猿来说,谁还在乎是男娃女娃,只要是自己的 ...
- java数据结构----堆
1.堆:堆是一种树,由它实现的优先级队列的插入和删除的时间复杂度都是O(logn),用堆实现的优先级队列虽然和数组实现相比较删除慢了些,但插入的时间快的多了.当速度很重要且有很多插入操作时,可以选择堆 ...
随机推荐
- Jenkins Android 自动打包配置
一.Jenkins自动打包配置 目标:1. 自动打包:2. 自动上传:3. 友好下载 1. Jenkins简介 Jenkins是基于Java开发的一种持续集成工具,用于监控持续重复的工作. 减少重复劳 ...
- PHP 常量dirname(__file__)
PHP 常量dirname(__FILE__)取得当前文件的绝对路径. define('ROOT_PATH', str_replace('includes/2.php', '', str_replac ...
- RFM模型及R语言实现
每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 一.基本概念 根据美国数据库营销研究所Arth ...
- STM32 下的库函数和寄存器操作比较
以 led闪烁中的flashLed函数例子: 库函数操作简单,但是效率不如寄存器操作的高: 寄存器操作很复杂,因为要熟悉上百个寄存器,但是程序效率很高 /**下面是通过直接操作库函数的方式实现IO控制 ...
- 如何访问pcie整个4k的配置空间
目前用于访问PCIe配置空间寄存器的方法需要追溯到原始的PCI规范.为了发起PCI总线配置周期,Intel实现的PCI规范使用IO空间的CF8h和CFCh来分别作为索引和数据寄存器,这种方法可以访问所 ...
- Netty未来展望
作为<Netty权威指南(第2版)>的结尾章节,和读者朋友们一起展望下Netty的未来. 1应用范围 随着大数据.互联网和云计算的发展,传统的垂直架构逐渐将被分布式.弹性伸缩的新架构替代. ...
- Linux显示PCI设备
Linux显示PCI设备 youhaidong@youhaidong-ThinkPad-Edge-E545:~$ lspci -tv -[0000:00]-+-00.0 Advanced Micro ...
- javaWeb事务
JDBC事务: cmd 命令上的事务开启: start transaction; / begin; 回滚 rollback; 提交 commit; JDBC事务控制: 开启事务:co ...
- 关于CS1061报错(XX不包含XXX的定义,并且找不到类型为XX的第一个参.....)的一种可能的解决的办法
在我编程中,我遇到了一个这样的报错, 可是我引用的product类中又确实定义了这么一个方法, protected void BindPageData(int categoryID) { Produc ...
- 第九篇:Map/Reduce 工作机制分析 - 作业的执行流程
前言 从运行我们的 Map/Reduce 程序,到结果的提交,Hadoop 平台其实做了很多事情. 那么 Hadoop 平台到底做了什么事情,让 Map/Reduce 程序可以如此 "轻易& ...