造轮子-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),用堆实现的优先级队列虽然和数组实现相比较删除慢了些,但插入的时间快的多了.当速度很重要且有很多插入操作时,可以选择堆 ...
随机推荐
- js将汉字转为相应的拼音
<!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8&quo ...
- 经典卷积神经网络(LeNet、AlexNet、VGG、GoogleNet、ResNet)的实现(MXNet版本)
卷积神经网络(Convolutional Neural Network, CNN)是一种前馈神经网络,它的人工神经元可以响应一部分覆盖范围内的周围单元,对于大型图像处理有出色表现. 其中 文章 详解卷 ...
- 1.2 PCI总线的信号定义
PCI总线是一条共享总线,在一条PCI总线上可以挂接多个PCI设备.这些PCI设备通过一系列信号与PCI总线相连,这些信号由地址/数据信号.控制信号.仲裁信号.中断信号等多种信号组成. PCI总线是一 ...
- win10系统搭建虚拟机:VMware Workstation Player 12环境+Ubuntu Kylin 16.04 LTS系统
笔者小白一枚,其实连虚拟机是个啥都不知道...实属惭愧,介于此所以今天倒腾了一下花了一上午就已经搭建好一个VMware Workstation Player 12免费版的,很哈皮,于是赶紧分享一下. ...
- 实战DeviceIoControl 之四:获取硬盘的详细信息
Q 用IOCTL_DISK_GET_DRIVE_GEOMETRY或IOCTL_STORAGE_GET_MEDIA_TYPES_EX只能得到很少的磁盘参数,我想获得包括硬盘序列号在内的更加详细的信息,有 ...
- DirectShow中写push模式的source filter流程 + 源代码(内附详细注释)
虽然网上已有很多关于DirectShow写source filter的资料,不过很多刚开始学的朋友总说讲的不是很清楚(可能其中作者省略了许多他认为简 单的过程),读者总希望看到象第一步怎么做,第二步怎 ...
- FFMPEG:H264解码-SDL显示(RGB32、RGB24、YUV420P、YUV422)
FFMpeg对视频文件进行解码的大致流程 1. 注册所有容器格式: av_register_all()2. 打开文件: av_open_input_file()3. 从文件中提取流信息: av_fin ...
- Python与RabbitMQ交互
RabbitMQ 消息队列 成熟的中间件RabbitMQ.ZeroMQ.ActiveMQ等等 RabbitMQ使用erlang语言开发,使用RabbitMQ前要安装erlang语言 RabbitMQ允 ...
- 存储过程 100w提交
create or replace procedure largedata_insert(ip_table_name in varchar2, --目标表 ip_table_column in v ...
- Struts2【OGNL、valueStack】就是这么简单
什么是OGNL表达式? OGNL是Object Graphic Navigation Language 是操作对象属性的开源表达式. Struts2框架使用OGNL作为默认的表达式语言. 为什么我们学 ...