文章目录

  • 优先级队列
  • 模拟实现优先级队列
    • 向下调整建堆
    • 向上调整建堆
    • 堆的删除
  • priorityQueue
    • 构造方法
    • 大根堆和小根堆的向上调整比较方法
    • 扩容
  • 面试题
  • 堆排序

优先级队列

priorityqueue:底层是一颗完全二叉树

  1. 小根堆:根比左右孩子都小
  2. 大根堆:根比左右孩子都大
  3. 用数组存储

在这里插入图片描述

模拟实现优先级队列

向下调整建堆

  1. 向下调整算法的时间复杂度:O(N)
    建堆的算法

在这里插入图片描述
2. 推导过程:

在这里插入图片描述

	// 向下调整算法public void shifDown(int parent,int len){// parent每次从根节点开始向下调整// usedSize来检测是否还有得调,是否调结束了int child = 2 * parent + 1;// 至少有右孩子while(child < len){// 左右孩子比较大小,如果右孩子大,那么child+1if(child + 1 < len && elem[child] < elem[child + 1]){child = child + 1;}// if语句走完,证明child是左右孩子中大的那个的下标if(elem[child] > elem[parent]){int tmp = elem[child];elem[child] = elem[parent];elem[parent] = tmp;parent = child;child = parent * 2 + 1;}else{// 证明左右孩子中最大的那个都比父亲节点小,// 是大根堆,不用调整了break;}}}

向上调整建堆

  1. 新插入一个节点并且向上调整为大根堆
  2. 向上调整建堆的时间复杂度是:O(N * logN)

在这里插入图片描述

 // 插入一个节点向上调整算法public void push(int val){// 满了,扩容if(isFull()){elem = Arrays.copyOf(elem,2 * elem.length);}elem[usedSize] = val;// 向上调整,usedSize为新插入元素的下标siftUp(usedSize);usedSize++;}public void swap(int i,int j){int tmp = elem[i];elem[i] = elem[j];elem[j] = tmp;}public boolean isFull(){return usedSize == elem.length;}public void siftUp(int child){// 通过孩子节点找到父亲节点下标// 只要child大于0还需要调整,=0就不需要调整了while(child > 0) {int parent = (child - 1) / 2;if (elem[parent] < elem[child]) {swap(child, parent);child = parent;parent = (child - 1) / 2;} else {// parent下标对应的元素大于child下标对应的元素// 不需要交换break;}}}

堆的删除

  1. 让堆顶元素和最后一个元素交换
  2. 然后让usedSize–,就删除了最后一个元素
  3. 最后只需要调整0下标这棵树就行了,使用向下调整算法
public int pop(){// 判空if(empty()){return -1;}int tmp = elem[0];swap(0,usedSize - 1);usedSize--;shifDown(0,usedSize);return tmp;}public boolean  empty(){return usedSize == 0;}

priorityQueue

  1. Java中的优先级队列默认是小根堆
public static void main(String[] args) {// 默认是小根堆PriorityQueue<Integer> priorityQueue = new PriorityQueue<>();priorityQueue.offer(1);priorityQueue.offer(5);priorityQueue.offer(6);System.out.println(priorityQueue.poll());// 1System.out.println(priorityQueue.poll());// 5}
  1. PriorityQueue必须存放可以比较大小的元素
  2. 不能插入null对象,否则会抛出空指针异常
  3. 没有容量限制,可以插入任意多个元素,其内部可以自动扩容,在堆上开空间的
  4. 插入和删除的时间复杂度都是O(logN)

构造方法

在这里插入图片描述

大根堆和小根堆的向上调整比较方法

  1. 插入元素,向上调整,向上调整的比较方法
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

扩容

  1. 要么2倍扩容,要么1.5倍扩容
    在这里插入图片描述

面试题

  1. top-k问题:
    解法一:
    比如得到最小的前k个元素
    建立一个小根堆
    出k次元素得到最小的前k个元素

解法二:
求最小的前k个元素,先把前k个元素建立大根堆,再和k+1位置的元素比较,如果小于堆顶元素就入堆,并且删除堆顶元素,以此类推,最后剩下的k个元素就是最小的元素
在这里插入图片描述
3. top-k问题的时间复杂度是:O(N * logK)
求最小的K个数

// class Imp implements Comparator<Integer> {
//     public int compare(Integer o1,Integer o2){
//         return o2.compareTo(o1);
//     }
// }class Solution {public int[] smallestK(int[] arr, int k) {int[] ret = new int[k];if(arr == null || k <= 0){return ret;}// new一个比较器,匿名内部类的方法PriorityQueue<Integer> priorityQueue = new PriorityQueue<>(new Comparator<Integer>(){public int compare(Integer o1,Integer o2){return o2.compareTo(o1);}});// 建立k个元素的大根堆// K * logKfor(int i = 0;i < k;i++){priorityQueue.offer(arr[i]);}// O((N-k) * logK)for(int i = k;i < arr.length;i++){int top = priorityQueue.peek();if(top > arr[i]){priorityQueue.poll();priorityQueue.offer(arr[i]);}}// 总的时间复杂度: O(N * logK)// K * logK// 整理元素不算入top-k问题中for(int i = 0;i < k;i++){ret[i] = priorityQueue.poll();}return ret;}
}

堆排序

  1. 从大到小或者是从小到大排序
  2. 从小到大排序,建立大根堆,每次最后一个元素和堆顶元素交换,usedSize–,向下调整为大根堆,以此类推
  3. 堆排序的时间复杂度:O(N * logN)

在这里插入图片描述

// 堆排序public void heapSort(){int end = usedSize - 1;while(end > 0){swap(0,end);shifDown(0,end);end--;}}public static void main(String[] args) {TestHeap testHeap = new TestHeap();int array[] = {27,15,19,18,28,34,65,49,25,37};testHeap.initElem(array);// 向下调整建堆:O(N)testHeap.createHeap();System.out.println("======");// O(N * logN)testHeap.heapSort();System.out.println("======");}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/90220.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/90220.shtml
英文地址,请注明出处:http://en.pswp.cn/bicheng/90220.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在.NET Core API 微服务中使用 gRPC:从通信模式到场景选型

目录 一、gRPC 基础&#xff1a;为什么它适合微服务&#xff1f; 二、gRPC 的四种通信模式及.NET Core 实现 1. 一元 RPC&#xff08;Unary RPC&#xff09;&#xff1a;最基础的请求 - 响应模式 2. 服务器流式 RPC&#xff08;Server Streaming RPC&#xff09;&#xff1…

HTML零基础快速入门教程(详细篇)

本文详细介绍HTML零基础快速入门的基础知识&#xff0c;包括HTML的介绍、语言的一些实际作用、语法规范注意&#xff0c;如标签结构、标签属性、大小写不敏感等&#xff0c;还介绍了HTML文件的具体书写规则&#xff0c;如文件扩展名、文档类型声明和HTML结构以及具体的一些HTML…

LLM评测框架Ragas:SQL指标(解决了Ollama推理框架不支持的问题)

SQL类的度量指标是指运行SQL后的结果和预期之间的一个度量值。 datacompy score datacompy score 使用DataCompy(一个比较pandas的数据格式的python类,所以需要按照datacompy:pip install datacompy),默认是按照rows比较,也可以设置按照columns比较,这个事通过mode参数…

ubuntu24 ros2 jazzy

安装2 software & update 选择其它 安装 一、前提准备 检查操作系统版本&#xff1a; 确保你的系统版本是Ubuntu 24.04。你可以通过运行lsb_release -a命令来检查当前的系统版本。 设置UTF-8支持&#xff1a; ROS 2需要UTF-8编码支持。你可以通过以下命令来检查和设置UTF…

设备虚拟化技术

设备虚拟化技术概述设备虚拟化技术通过软件模拟物理硬件设备&#xff0c;使多个操作系统或应用程序能够共享同一台物理设备。它广泛应用于云计算、服务器整合和测试环境等领域。核心目标是提高资源利用率、隔离性和灵活性。•当接入的用户数增加到原交换机端口密度不能满足接入…

开发避坑短篇(3):解决@vitejs plugin-vue@5.0.5对Vite^5.0.0的依赖冲突

异常信息 # npm resolution error reportWhile resolving:system3.8.8 Found: vite6.2.3 node_modules/vitedev vite"6.2.3" from the root projectCould not resolve dependency: peer vite"^5.0.0" from vitejs/plugin-vue5.0.5 node_modules/vitejs/plu…

k8s快速部署(亲测无坑)

文章目录k8s快速部署&#xff08;亲测无坑&#xff09;一、网络划分二、CentOS7设置 标题固定IP和阿里云YUM源三、主机环境配置四、虚拟机的拷贝五、安装docker(每台主机都需要安装)六、安装kubelet,kubeadm,kubectl(每台机器都需要执行)遇到的问题参考文档k8s快速部署&#xf…

简易RAG问答引擎的构建与体验

RAG&#xff08;检索增强生成&#xff09;是结合检索与生成式 AI 的技术框架。核心逻辑是先从外部知识库精准检索相关信息&#xff0c;再将其作为上下文输入大模型生成回答。技术上依赖检索引擎&#xff08;如向量数据库、BM25&#xff09;、大语言模型&#xff08;如 GPT、LLa…

C++11特性学习 Day1

nullptr对于c中null (void*)0&#xff0c;所以在为函数传参传入0时&#xff0c;无法清楚地分辨是int类型的0还是指的是空指针null在C11中清晰的将空指针变为了nullptr&#xff0c;0专指int型的数字0override关键字在子类中对父类的函数的覆写之后加上override关键字&#xff0…

微算法科技(NASDAQ: MLGO)探索优化量子纠错算法,提升量子算法准确性

随着量子计算技术的飞速发展&#xff0c;量子计算机在解决复杂计算问题上的潜力日益显现。然而&#xff0c;量子计算面临的一个重大挑战是量子比特的脆弱性&#xff0c;即量子比特容易受到环境噪声和干扰的影响&#xff0c;导致量子态的塌缩和计算结果的错误。微算法科技&#…

MongoDB数据库详解-针对大型分布式项目采用的原因以及基础原理和发展-卓伊凡|贝贝|莉莉

MongoDB数据库详解-针对大型分布式项目采用的原因以及基础原理和发展-卓伊凡|贝贝|莉莉由于老产品即时通讯私有化软件就是采用MongoDB &#xff0c;但是版本实在太低&#xff0c;要做大更新&#xff0c;其次针对10年前完美运营的项目来到10年后的现在就不一定行&#xff0c;优雅…

Kotlin 中的单例模式(Singleton)与对象声明

在 Kotlin 中&#xff0c;类描述的是一种通用结构&#xff0c;可以多次实例化&#xff0c;也可以用多种方式实例化。但有时我们只需要单个实例&#xff0c;不多不少。单例模式能帮你更好地组织代码&#xff0c;把相关的方法聚合在一起。 单例模式是什么&#xff1f; 单例模式是…

Shell 编程基础入门从认识到实战

对于刚接触 Linux 或 Unix 系统的开发者来说&#xff0c;Shell 脚本往往是自动化操作的第一道门槛。它不像 Python 那样语法简洁&#xff0c;也不像 Java 那样有完善的面向对象体系&#xff0c;但却能以极少的代码实现强大的系统管理功能。本文将从 Shell 的基本概念讲起&#…

混合遗传粒子群算法在光伏系统MPPT中的应用研究

混合遗传粒子群算法在光伏系统MPPT中的应用研究 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家&#xff0c;觉得好请收藏。点击跳转到网站。 摘要 本文针对光伏系统最大功率点跟踪(MPPT)问题&#xff0…

机器视觉的布料丝印应用

在纺织印染行业&#xff0c;布料丝印工艺的精度直接决定产品外观质量与市场竞争力。传统丝印设备依赖机械定位与人工校准&#xff0c;面对高密度图案、柔性面料或复杂纹理时&#xff0c;易出现套色偏移、油墨渗透不均等问题&#xff0c;导致良品率波动与生产成本攀升。 随着机…

前端常用类库

常用类库 类库作用 类库可以帮助我们快速实现项目业务的开发与功能的实现, 帮助我们解放劳动力提高生产效率, 前端中的类库与框架都是由原生javascript编写, 提供给其他开发者应用于某一业务环境或者需求。一般有开发者/团队开源维护. 优秀的类库需要具备高度封装可用, 稳定, …

通俗易懂循环神经网络(RNN)指南

本文用直观类比、图表和代码&#xff0c;带你轻松理解RNN及其变体&#xff08;LSTM、GRU、双向RNN&#xff09;的原理和应用。什么是循环神经网络 循环神经网络&#xff08;Recurrent Neural Network, RNN&#xff09;是一类专门用于处理序列数据的神经网络。与前馈神经网络不同…

【SVM】支持向量机实例合集

基于Java的SVM(支持向量机)实例合集 以下是一个基于Java的SVM(支持向量机)实例合集,包含核心代码示例和应用场景说明。这些例子基于流行的机器学习库(如LIBSVM、Weka、JSAT)实现。 数据准备与加载 使用LIBSVM格式加载数据集: // 加载LIBSVM格式数据 svm_problem pr…

Python100个库分享第38个—lxml(爬虫篇)

目录专栏导读&#x1f4da; 库简介&#x1f3af; 主要特点&#x1f6e0;️ 安装方法Windows安装Linux/macOS安装验证安装&#x1f680; 快速入门基本使用流程HTML vs XML解析&#x1f50d; 核心功能详解1. XPath选择器2. CSS选择器支持3. 元素操作&#x1f577;️ 实战爬虫案例…

imx6ull-系统移植篇17——linux顶层 Makefile(上)

目录 前言 顶层 Makefile 源码简析 版本号 MAKEFLAGS 变量 命令输出 静默输出 设置编译结果输出目录 代码检查 模块编译 设置目标架构和交叉编译器 调用 scripts/Kbuild.include 文件 交叉编译工具变量设置 头文件路径变量 导出变量 make xxx_defconfig 过程 …