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

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

相关文章

Python中使用uv创建环境及原理举例详解

《Python中使用uv创建环境及原理举例详解》uv是Astral团队开发的高性能Python工具,整合包管理、虚拟环境、Python版本控制等功能,:本文主要介绍Python中使用uv创建环境及... 目录一、uv工具简介核心特点:二、安装uv1. 通过pip安装2. 通过脚本安装验证安装:配置镜像源(可

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

Java 枚举的基本使用方法及实际使用场景

《Java枚举的基本使用方法及实际使用场景》枚举是Java中一种特殊的类,用于定义一组固定的常量,枚举类型提供了更好的类型安全性和可读性,适用于需要定义一组有限且固定的值的场景,本文给大家介绍Jav... 目录一、什么是枚举?二、枚举的基本使用方法定义枚举三、实际使用场景代替常量状态机四、更多用法1.实现接

Python使用Tkinter打造一个完整的桌面应用

《Python使用Tkinter打造一个完整的桌面应用》在Python生态中,Tkinter就像一把瑞士军刀,它没有花哨的特效,却能快速搭建出实用的图形界面,作为Python自带的标准库,无需安装即可... 目录一、界面搭建:像搭积木一样组合控件二、菜单系统:给应用装上“控制中枢”三、事件驱动:让界面“活”

Macos创建python虚拟环境的详细步骤教学

《Macos创建python虚拟环境的详细步骤教学》在macOS上创建Python虚拟环境主要通过Python内置的venv模块实现,也可使用第三方工具如virtualenv,下面小编来和大家简单聊聊... 目录一、使用 python 内置 venv 模块(推荐)二、使用 virtualenv(兼容旧版 P

如何确定哪些软件是Mac系统自带的? Mac系统内置应用查看技巧

《如何确定哪些软件是Mac系统自带的?Mac系统内置应用查看技巧》如何确定哪些软件是Mac系统自带的?mac系统中有很多自带的应用,想要看看哪些是系统自带,该怎么查看呢?下面我们就来看看Mac系统内... 在MAC电脑上,可以使用以下方法来确定哪些软件是系统自带的:1.应用程序文件夹打开应用程序文件夹

Linux lvm实例之如何创建一个专用于MySQL数据存储的LVM卷组

《Linuxlvm实例之如何创建一个专用于MySQL数据存储的LVM卷组》:本文主要介绍使用Linux创建一个专用于MySQL数据存储的LVM卷组的实例,具有很好的参考价值,希望对大家有所帮助,... 目录在Centos 7上创建卷China编程组并配置mysql数据目录1. 检查现有磁盘2. 创建物理卷3. 创

Java Stream.reduce()方法操作实际案例讲解

《JavaStream.reduce()方法操作实际案例讲解》reduce是JavaStreamAPI中的一个核心操作,用于将流中的元素组合起来产生单个结果,:本文主要介绍JavaStream.... 目录一、reduce的基本概念1. 什么是reduce操作2. reduce方法的三种形式二、reduce

Python Flask 库及应用场景

《PythonFlask库及应用场景》Flask是Python生态中​轻量级且高度灵活的Web开发框架,基于WerkzeugWSGI工具库和Jinja2模板引擎构建,下面给大家介绍PythonFl... 目录一、Flask 库简介二、核心组件与架构三、常用函数与核心操作 ​1. 基础应用搭建​2. 路由与参

Spring Boot中的YML配置列表及应用小结

《SpringBoot中的YML配置列表及应用小结》在SpringBoot中使用YAML进行列表的配置不仅简洁明了,还能提高代码的可读性和可维护性,:本文主要介绍SpringBoot中的YML配... 目录YAML列表的基础语法在Spring Boot中的应用从YAML读取列表列表中的复杂对象其他注意事项总