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

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

相关文章

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

zoj3820(树的直径的应用)

题意:在一颗树上找两个点,使得所有点到选择与其更近的一个点的距离的最大值最小。 思路:如果是选择一个点的话,那么点就是直径的中点。现在考虑两个点的情况,先求树的直径,再把直径最中间的边去掉,再求剩下的两个子树中直径的中点。 代码如下: #include <stdio.h>#include <string.h>#include <algorithm>#include <map>#

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

【区块链 + 人才服务】可信教育区块链治理系统 | FISCO BCOS应用案例

伴随着区块链技术的不断完善,其在教育信息化中的应用也在持续发展。利用区块链数据共识、不可篡改的特性, 将与教育相关的数据要素在区块链上进行存证确权,在确保数据可信的前提下,促进教育的公平、透明、开放,为教育教学质量提升赋能,实现教育数据的安全共享、高等教育体系的智慧治理。 可信教育区块链治理系统的顶层治理架构由教育部、高校、企业、学生等多方角色共同参与建设、维护,支撑教育资源共享、教学质量评估、

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定

AI行业应用(不定期更新)

ChatPDF 可以让你上传一个 PDF 文件,然后针对这个 PDF 进行小结和提问。你可以把各种各样你要研究的分析报告交给它,快速获取到想要知道的信息。https://www.chatpdf.com/