堆(逐个添加元素创建堆)-堆结构的创建+堆结构的实际应用

2024-09-07 01:44

本文主要是介绍堆(逐个添加元素创建堆)-堆结构的创建+堆结构的实际应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、问题引入

一堆数据的值不断变化,我们想要从这堆数据中获取到最大值或者最小值(比如电影排行榜中电影的排名,每部电影观看人数越多,那么该部电影的排名就越高)

1、数据结构解决的是数据的存储问题,算法解决的是存储了的数据的使用效率问题

2、数据结构里面的堆结构(heap)和Java语言中的数据堆不一样

3、数据结构里面的堆结构本质上是数组

4、数组的元素查询效率高(由于使用索引查询),时间复杂度为O(1),元素删除效率低(由于删除一个元素需要移动很多其他的元素)

5、二叉树的元素删除效率高(由于只需逐级调整),删除元素的时间复杂度为O(log_{2}^{n})

6、二叉树(平衡二叉树、满二叉树、完全二叉树)

完全二叉树:深度为h的二叉数,除到底h层外,其他各层(第1层到第(h-1)层)的结点数都达到最大数目,且第h层所有的结点都连续集中在最左边,具备这两个特点的二叉树就是完全二叉树

二、认识堆

1、堆的定义

(1)堆(heap)是计算机科学中一类特殊的数据结构的统称

(2)堆是一个可以被看作一棵树的数组对象(即将数组元素以特殊完全二叉树的形式在数组中进行存储的,特殊在子父结点在数组中的索引会满足一定的数学关系/即元素在数组中存储,但以完全二叉树结点的规律进行使用)

(3)堆的本质就是将数组元素转换为特殊完全二叉树的形式在数组中实现存储

(4)构建堆的本质就是利用堆的方式将数组中的元素在数组中进行重新排列

(5)堆最终是为优先队列准备的

2、堆的分类

(1)最大堆:根结点最大的堆,又称大根堆

(2)最小堆:根结点最小的堆,又称小根堆

3、堆的特点

(1)堆中某个结点(即元素)的值总是不大于(大根堆)或不小于(小根堆)其父结点的值

(2)一个堆可以等效于一棵完全二叉树(将堆数组中的元素从前向后按照完全二叉树的形式从上向下从左到右进行排列即可得到堆数组等效的完全二叉树)

4、完全二叉树父结点与子结点在堆数组中索引间的关系

(1)左子结点索引=父结点索引*2+1

(2)右结点索引=父结点索引*2+2

(3)父结点索引=(子结点索引-1)/2(除根结点外,根结点没有父结点)

三、创建堆

1、泛型:因为堆中的数据的类型可能为复杂数据类型,因此在创建堆时要使用泛型来规范堆中存储的数据的类型

2、新元素添加要比较:在向堆中填加新元素时必须要比较大小,因此在定义泛型时要扩展泛型,使得堆中的数据允许进行比较

public class MyHeap <T extends Comparable<T>>{}

3、泛型在通过new赋值时必须使用给定类型来new

elements=(T[])new Comparable[DEFAULT_SIZE];

4、最大堆的效率分析

(1)获取堆顶元素的时间复杂度为O(1)

(2)向堆中添加元素的时间复杂度为O(h)=O(log_{2}^{n})

三、堆的核心代码

package com.ffyc.heap;public class MyHeap <T extends Comparable<T>>{private final int DEFAULT_SIZE = 10;// 使用数组作为存储结构的数据结构数组在初始时必须有一个默认长度private T[] elements;private int size;// 堆的容量大小(堆实际存储的元素的数目),数据结构删除元素并没有真正删除,只是让删除元素无法被访问到public MyHeap(){elements=(T[])new Comparable[DEFAULT_SIZE];// 向下转型}private int getParentIndex(int index){return index==0?-1:(index-1)/2;}private int getLeftIndex(int index){return index*2+1;}private void swap(int i,int j){T tmp=elements[i];elements[i]=elements[j];elements[j]=tmp;}/*** 上浮:将元素放在最后一个位置,不停地逐级向上进行比较,如果比父元素大,则与父元素交换位置,直至比到不能比的时候,将元素放在合适的位置上* @param index 要上浮结点的索引*/private void floatUp(int index){// 获得父结点的索引int parentIndex=getParentIndex(index);while (parentIndex != -1&&elements[index].compareTo(elements[parentIndex])>0){swap(index, parentIndex);// 迭代法index = parentIndex;parentIndex=getParentIndex(index);}}// 添加数据public void offer(T data){elements[size++]=data;// 上浮问题 调整末尾元素的位置floatUp(size-1);}// 判断堆是否为空public boolean isEmpty(){return size==0;}// 获得堆顶元素public T peek(){if(isEmpty())return null;return elements[0];}@Overridepublic String toString() {StringBuilder sb=new StringBuilder("[");for (int i = 0; i < size; i++) {sb.append(elements[i]+",");}sb.deleteCharAt(sb.length()-1);sb.append("]");return sb.toString();}private void swimDown(int parentIndex){T swimData=elements[parentIndex];int currentIndex=parentIndex;int leftIndex=getLeftIndex(currentIndex);while(leftIndex<size){int rightIndex=leftIndex+1;int highIndex=leftIndex;// 子结点的最大值的索引if(leftIndex < size && elements[highIndex].compareTo(elements[rightIndex])<0){highIndex=rightIndex;}if(swimData.compareTo(elements[highIndex])<0){swap(currentIndex, highIndex);currentIndex=highIndex;leftIndex=getLeftIndex(currentIndex);}else{break;}}}// 删除堆顶元素public T poll(){if(isEmpty()) return null;// 交换堆顶元素和最后一个元素的位置(使用最后一个元素覆盖堆顶元素)T data=elements[0];elements[0]=elements[--size];// 下沉问题 调整堆顶元素的位置swimDown(0);return data;}}

四、堆的应用

1、从一堆数据中获取第二大数据但至多只能用一次while循环(微软面试题,不能用Arrays类中的sort方法)

public static void main(String[] args) {MyHeap<Integer>myHeap=new MyHeap<>();myHeap.offer(3);myHeap.offer(7);myHeap.offer(5);myHeap.offer(8);myHeap.offer(2);myHeap.offer(4);myHeap.offer(1);myHeap.poll();System.out.println(myHeap.peek());}

2、获取所有新闻中点击量最大的新闻的id

package com.ffyc.heap;public class News implements Comparable<News>{private int id;// 新闻的唯一标识private int count;// 新闻的点击量public News(int id, int count) {this.id = id;this.count = count;}public int getId() {return id;}public void setId(int id) {this.id = id;}public int getCount() {return count;}public void setCount(int count) {this.count = count;}@Overridepublic int compareTo(News o) {return this.count-o.getCount();}public static void main(String[] args) {MyHeap<News>myHeap=new MyHeap<>();myHeap.offer(new News(1, 15));myHeap.offer(new News(2, 25));myHeap.offer(new News(3, 13));myHeap.offer(new News(4, 12));myHeap.offer(new News(5, 27));// 获得新闻排行榜点击量最高的新闻的idSystem.out.println(myHeap.peek().getId());// 将新闻排行榜点击量最高的新闻的id下架myHeap.poll();System.out.println(myHeap.peek().getId());}
}

五、堆的缺点(通过逐个添加元素创建的堆)

无法实现改变其中一个新闻的点击量后实现所有新闻的重新排名(根据点击量的大小进行排名)

这篇关于堆(逐个添加元素创建堆)-堆结构的创建+堆结构的实际应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1143701

相关文章

线程池ThreadPoolExecutor应用过程

《线程池ThreadPoolExecutor应用过程》:本文主要介绍如何使用ThreadPoolExecutor创建线程池,包括其构造方法、常用方法、参数校验以及如何选择合适的拒绝策略,文章还讨论... 目录ThreadPoolExecutor构造说明及常用方法为什么强制要求使用ThreadPoolExec

mysql_mcp_server部署及应用实践案例

《mysql_mcp_server部署及应用实践案例》文章介绍了在CentOS7.5环境下部署MySQL_mcp_server的步骤,包括服务安装、配置和启动,还提供了一个基于Dify工作流的应用案例... 目录mysql_mcp_server部署及应用案例1. 服务安装1.1. 下载源码1.2. 创建独立

C#高效实现在Word文档中自动化创建图表的可视化方案

《C#高效实现在Word文档中自动化创建图表的可视化方案》本文将深入探讨如何利用C#,结合一款功能强大的第三方库,实现在Word文档中自动化创建图表,为你的数据呈现和报告生成提供一套实用且高效的解决方... 目录Word文档图表自动化:为什么选择C#?从零开始:C#实现Word文档图表的基本步骤深度优化:C

Python列表的创建与删除的操作指南

《Python列表的创建与删除的操作指南》列表(list)是Python中最常用、最灵活的内置数据结构之一,它支持动态扩容、混合类型、嵌套结构,几乎无处不在,但你真的会创建和删除列表吗,本文给大家介绍... 目录一、前言二、列表的创建方式1. 字面量语法(最常用)2. 使用list()构造器3. 列表推导式

MySQL快速复制一张表的四种核心方法(包括表结构和数据)

《MySQL快速复制一张表的四种核心方法(包括表结构和数据)》本文详细介绍了四种复制MySQL表(结构+数据)的方法,并对每种方法进行了对比分析,适用于不同场景和数据量的复制需求,特别是针对超大表(1... 目录一、mysql 复制表(结构+数据)的 4 种核心方法(面试结构化回答)方法 1:CREATE

JavaWeb项目创建、部署、连接数据库保姆级教程(tomcat)

《JavaWeb项目创建、部署、连接数据库保姆级教程(tomcat)》:本文主要介绍如何在IntelliJIDEA2020.1中创建和部署一个JavaWeb项目,包括创建项目、配置Tomcat服务... 目录简介:一、创建项目二、tomcat部署1、将tomcat解压在一个自己找得到路径2、在idea中添加

Java利用Spire.Doc for Java实现在模板的基础上创建Word文档

《Java利用Spire.DocforJava实现在模板的基础上创建Word文档》在日常开发中,我们经常需要根据特定数据动态生成Word文档,本文将深入探讨如何利用强大的Java库Spire.Do... 目录1. Spire.Doc for Java 库介绍与安装特点与优势Maven 依赖配置2. 通过替换

Nginx内置变量应用场景分析

《Nginx内置变量应用场景分析》Nginx内置变量速查表,涵盖请求URI、客户端信息、服务器信息、文件路径、响应与性能等类别,这篇文章给大家介绍Nginx内置变量应用场景分析,感兴趣的朋友跟随小编一... 目录1. Nginx 内置变量速查表2. 核心变量详解与应用场景3. 实际应用举例4. 注意事项Ng

java创建xls文件放到指定文件夹中实现方式

《java创建xls文件放到指定文件夹中实现方式》本文介绍了如何在Java中使用ApachePOI库创建和操作Excel文件,重点是如何创建一个XLS文件并将其放置到指定文件夹中... 目录Java创建XLS文件并放到指定文件夹中步骤一:引入依赖步骤二:创建XLS文件总结Java创建XLS文件并放到指定文件

Java中的随机数生成案例从范围字符串到动态区间应用

《Java中的随机数生成案例从范围字符串到动态区间应用》本文介绍了在Java中生成随机数的多种方法,并通过两个案例解析如何根据业务需求生成特定范围的随机数,本文通过两个实际案例详细介绍如何在java中... 目录Java中的随机数生成:从范围字符串到动态区间应用引言目录1. Java中的随机数生成基础基本随