【C++】详解STL的适配器容器之一:优先级队列 priority_queue

目录

堆算法

概述

向下调整建堆

向上调整建堆

建堆算法

仿函数

概述

使用介绍

emtpy

size

top

push

pop

模拟实现

仿函数

框架

向下调整算法

向上调整算法

pop

push

empty

top


要理解优先级队列,需要有如下知识

STL容器之一的vector,小编写了写了五千字长文详解了vector容器,不过大家只需要知道vector是什么即可http://t.csdnimg.cn/tz9y6

堆算法,虽然小编在学C语言的时候写过一篇,但本篇内容会详细讲解堆算法

仿函数,仿函数属于STL六大组件之一,小编也会精讲

堆算法

概述

小编在学习C语言时写过一篇堆排序,详见http://t.csdnimg.cn/pT5Vw

堆在结构上是一颗二叉树,这颗二叉树只能是满二叉树或完全二叉树。这颗树上的所有数据存放在类似于数组的顺序表中,用顺序表来管理树的数据。(顺序表是一种数据结构,它的底层是线性的空间——存储数据的空间是连续的)

树上的数据是按层序的顺序存入顺序表。如下图

那么顺序表的下标就代表树的节点。父亲节点,左孩子节点,右孩子节点的下标关系如下

左孩子节点的下标等于父亲节点的下标乘2加1
右孩子节点的下标等于左孩子节点的下标加1
父亲节点的下标等于左孩子节点的下标减1除2
左孩子节点的下标等于右孩子节点的下标减1

到这里大家也看出来了,我们所谓的树结构只是想象的实际是我们管理的只是类似于数组的顺序表通过上述公式便可以达到顺序表是一颗树形结构的效果

为什么非要搞一颗树形结构呢

实际上,只用用树形结构存储数据的话,和顺序表,链表比是没有任何优势的。如果存储数据时加上某些限制,便可以高效的对数据进行排序,查找等。

本来顺序表的排序效率是O(N^2),但如果顺序表管理的是一颗树形结构,那么它的排序效率会被降到O(N * lgN)。O(N * lgN)的效率在所有排序效率中属于第一梯队

那么在堆存储数据时,存数据时的限制是:父亲节点存入的数据要大于或小于孩子节点存入的数据,如果是大于就是大堆,如果是小于就是小堆

大堆特性:堆顶的数据是最大的,因为堆顶的节点是所有节点的父亲节点,而存数据时父亲节点存入的数据要大于孩子节点存入的数据,所以堆顶的数据是最大的。

小堆特性:堆顶的树据是最小的,原因同上。

向下调整建堆

如果某一个数据使堆不符合堆的结构,便可以使用向下调整算法。核心逻辑如下

恢复成小堆

从父亲节点开始

比较出左孩子节点和右孩子节点较小的节点

父亲节点是否大于孩子节点

是:交换父亲节点和孩子节点,并继续比较

否:终止整个逻辑

时间复杂度分析:

交换数据最坏的情况下是交换高度次——lgN次。效率是O(lgN)

向上调整建堆

与向上调整建堆相似,也是一种恢复堆结构的算法。

它是从孩子节点开始,向上比较父亲节点。向上调整建堆是不用比较左右孩子的。每个孩子节点有且只有一个父亲节点

时间复杂度分析:

交换数据最坏的情况下是交换高度次——lgN次。效率是O(lgN)

建堆算法

如果顺序表中的数据不是一个数据不符合堆结构,而是所有或大部分数据都不符合堆结构呢

建堆算法就是让顺序表中的无序的数据按照堆结构排序,使顺序表符合堆的结构。

核心逻辑非常简单:

最后一个父亲节点开始往前遍历每一个节点,每遍历一个节点就调用一次向下调整算法

最后一个父亲节点的下标等于最后一个孩子节点的下标减1除2

时间复杂度

关于建堆的时间复杂度参考小编的另一篇文章

不推公式,形象理解堆排序的时间复杂度:

http://t.csdnimg.cn/HfH1G

仿函数

仿函数是C++的STL的六大组件之一

从名字上理解,它具有函数的功能,但不是函数。如果从实现的角度讲的话,叫函数对象比较贴切——有函数功能的对象

再通俗一点,就是一个里对 ( ) 进行了运算符重载,不清楚运算符重载的话可参考http://t.csdnimg.cn/1NksT

重载之后,该实例化的对象即可像函数以样调用,不管从写法还是效果上,与函数无异。如下示意

template <class T>  //仿函数 比较是否大于  模板
class Less  
{
public:bool operator()(const T& x, const T& y)   {return x < y; //如果 x y是自定义类型,那么 < 便是自定义类型的赋值重载(前提是自定义类型支持 < 的赋值重载)}
};

Less less;//实例化对象

int i = less(3, 5);//像函数一样调用

仿函数的出现是为了代替函数指针。首先函数指针的定义是很繁琐的,可读性极差(虽然可以定义别名)。整个STL的设计都是泛型编程,指针很死板,不适合泛型编程。

概述

到此,所有的知识铺垫完毕,那么什么是优先级队列呢,如下图示意

优先级队列实际上还是堆。

vector容器承担顺序表这一数据结构,堆算法负责管理vector容器中的数据,仿函数是为了控制大堆和小堆

优先级队列不提供迭代器,也就是说优先级队列不支持元素遍历。

优先级队列核心功能是入数据和出数据。入数据优先级队列会调用向下调整算法或向上调整算法建堆,出数据会只会出最值

使用介绍

emtpy

判断队列是否为空

size

返回队列数据个数

top

返回堆顶数据

push

入数据

pop

删除数据

模拟实现

仿函数

template <class T>  //仿函数 比较是否小于
class Less  
{
public:bool operator()(const T& x, const T& y)   {return x < y; //如果 x y是自定义类型,那么 < 便是自定义类型的赋值重载(前提是自定义类型支持 < 的赋值重载)}
};template <class T> //仿函数  比较是否大于
class Greater
{public:bool operator()(const T& x, const T& y)     {return x > y;}
};

类里的成员函数一定要设计成共有,因为优先级队列要访问。

框架

template <class T, class  Container = std::vector<T>, class Comapre = Less<T>> 
//三个模板参数,<类型, 容器, 仿函数> class priorit_queue
{ //......private:Container con;//容器};

向下调整算法

//向下调整建堆
void AdjustDown(size_t father)
{//私有接口,不需要检查坐标的合法性Comapre compare; //实例化对象 ,比较大于还是小于传仿函数即可     size_t child = father * 2 + 1; //左孩子的坐标  while (child < con.size()) {if (child + 1 < con.size() && compare(con[child], con[child + 1])) {child += 1;}if (compare(con[father], con[child])){std::swap(con[father], con[child]); //交换两个节点  father = child;   //更改下标child = father * 2 + 1;}else{break; //终止循环}}
};

向上调整算法

void AdjustUp(size_t child)
{Comapre compare; //实例化对象      size_t father = (child - 1) / 2;while (child > 0) {if (compare(con[father], con[child])){std::swap(con[father], con[child]);child = father;father = (child - 1) / 2;}else{break;}}}

向上调整算法和向下调整算法设计成私有即可

pop

void pop() //删除数据
{std::swap(con[0], con[con.size() - 1]); //首尾交换   con.pop_back(); //删除尾部数据AdjustDown(0);//重新建堆
};

push

void push(const T& x) //插入数据
{con.push_back(x);  //尾插AdjustUp(con.size() - 1); //向上调整算法
};

empty

bool empty() const //判断是否为空
{return con.size() == 0;
}

top

const T& top() const  //返回堆顶元素
{return con[0];
}

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

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

相关文章

聚类分析 | 基于GA遗传算法优化kmeans聚类(Matlab)

聚类分析 | 基于GA遗传算法优化kmeans聚类&#xff08;Matlab&#xff09; 目录 聚类分析 | 基于GA遗传算法优化kmeans聚类&#xff08;Matlab&#xff09;效果一览基本介绍程序设计参考资料 效果一览 基本介绍 GA-kmeans聚类算法&#xff0c;通过GA遗传算法优化kmeans聚类&…

序列化的不同格式:JSON、XML、TOML、CSON、YAML

前言 这篇文章参考于知乎&#xff0c;进行了一些总结。 正文 首先什么是序列化&#xff0c;数据序列化是从一个系统获取一些信息&#xff0c;将其转换为其它系统可以读取的格式&#xff0c;然后将其传递给其它系统的过程。也就是可以让不同系统“通信”。 序列化需要满足两…

JetPack Compose Navigation

1. 导入依赖 implementation("androidx.navigation:navigation-compose:2.7.7") 2.kotlin编译版本升级 composeOptions {kotlinCompilerExtensionVersion "1.5.0"} 3.插件版本升级 // Top-level build file where you can add configuration options c…

学习笔记:IEEE 1003.13-2003【POSIX PSE51接口列表】

一、POSIX PSE51接口列表 根据IEEE 1003.13-2003&#xff0c;整理了POSIX PSE51接口API&#xff08;一共286个&#xff09;&#xff0c;每个API支持链接查看。 IEEE POSIX接口online搜索链接&#xff1a; The Open Group Base Specifications Issue 7, 2018 edition 详细内…

【python】模块与包

Python中的模块和包是组织和管理代码的重要工具。通过模块和包&#xff0c;你可以更好地管理和重用你的代码&#xff0c;使得代码更加模块化和可维护。 目录 前言 正文 一、模块 1、模块的分类 1&#xff09;内置模块 python解释器中默认拥有的模块可以直接使用&#xff08;…

用户需求甄别和筛选的6大标准

产品经理日常经常接收到大量的需求&#xff0c;并不是所有的需求都需要开发&#xff0c;需要进行甄别和筛选&#xff0c;这样有利于确保项目的成功、优化资源利用以及提高产品质量。 那么针对这些用户需求进行甄别或筛选的评判标准是什么&#xff1f;需求筛选可以说是初步的需求…

代码随想录-算法训练营day31【贪心算法01:理论基础、分发饼干、摆动序列、最大子序和】

代码随想录-035期-算法训练营【博客笔记汇总表】-CSDN博客 第八章 贪心算法 part01● 理论基础 ● 455.分发饼干 ● 376. 摆动序列 ● 53. 最大子序和 贪心算法其实就是没有什么规律可言&#xff0c;所以大家了解贪心算法 就了解它没有规律的本质就够了。 不用花心思去研究其…

C++牛客周赛题目分享(2)小红叕战小紫,小红的数组移动,小红的素数合并,小红的子序列求和

目录 ​编辑 1.前言 2.四道题目 1.小红叕战小紫 1.题目描述 2.输入描述 3.输出描述 4.示例 5.题解与思路 2.小红的数组移动 1.题目描述 2.输入描述 3.输出描述 4.示例 5.题解与思路 3.小红的素数合并 1.题目描述 2.输入描述 3.输出描述 4.示例 5.题解与思…

增强For循环执行顺序探究

增强For循环执行顺序探究 增强For循环基础执行顺序探讨对于数组对于集合 经典示例数组示例集合示例&#xff08;ArrayList&#xff09; 注意事项结论 在Java编程中&#xff0c;增强型for循环&#xff08;也称为“foreach”循环&#xff09;是一种简洁而强大的迭代集合或数组元素…

super

super 一、理解 super.属性&#xff1a;在子类中&#xff0c;调用父类非私有化的成员属性 super.方法&#xff1a;在子类中&#xff0c;调用父类非私有化的成员方法 super()&#xff1a;在子类构造方法中调用父类非私有的构造方法 二、案例 需求&#xff1a;编写中国人和日本人…

云原生新手和开源教育分论坛 01-Kubernetes 社区:从新手到影响者

2024年04月20日 上海KCD & Shanghai&#xff1a;https://community.cncf.io/events/details/cncf-kcd-shanghai-presents-kcd-shanghai-2024/视频观看&#xff1a;https://www.bilibili.com/video/BV1nD421T786/?spm_id_from333.999.0.0&vd_sourceae7b192be069682aabc…

【FreeRTOS 快速入门】-- 1、STM32工程移植FreeRTOS

目录 一、新建STM32工程 为了示范完整的移植过程&#xff0c;我们从0开始&#xff0c;新建一个标准的STM32点灯工程。 &#xff08;本篇以CubeMX作示范&#xff0c;CubeIDE操作近同&#xff0c;可作对比参考&#xff09; 1、新建工程 选择 芯片型号 新建工程 2、搜索芯片型号…

24年做抖音小店,你还停留在数据?别人都已经开始注重利润了

大家好&#xff0c;我是电商笨笨熊 一件事情持续做&#xff0c;一个项目持续深耕&#xff0c;意义到底是什么&#xff1f; 这句话我常常说&#xff0c;但很多人似乎走偏了实际意义&#xff1b; 尤其对于新手来说&#xff0c;做抖音小店总是向往某某老玩家多么牛的数据&#…

程序员健康指南:运动,让代码更流畅

程序员健康指南&#xff1a;运动&#xff0c;让代码更流畅 程序员&#xff0c;一个与电脑相伴的群体&#xff0c;长时间的久坐和高强度的脑力劳动是他们的日常。然而&#xff0c;久坐不仅影响体态&#xff0c;更对心脏健康构成威胁。根据《欧洲心脏杂志》的研究&#xff0c;中…

第十三节 huggingface的trainner解读与Demo

文章目录 前言一、trainer和TrainingArguments训练与预测完整Demo1、数据构建2、TrainingArguments构建3、Trainer初始化4、模型训练5、模型推理6、完整demo代码7、完整运行结果二、辅助函数1、yield返回内容2、迭代器中断恢复迭代demo3、yield from结构4、torch.Generator()的…

【PPT技巧】ppt文件打开就是只读模式,如何改为可编辑模式?

PPT文档打开是只读模式&#xff0c;如何改成可编辑文档呢&#xff1f;这需要分几种情况来说&#xff0c;所以今天将介绍几种方法帮助PPT只读文档改为可编辑文档。 方法一&#xff1a; 我们可以先查看一下文件属性&#xff0c;属性中有只读属性&#xff0c;当我们打开文档之后带…

C++入门——引用(2)

前言 上一节我们开始学习了C&#xff0c;并且对C有了初步的了解&#xff0c;这一节我们继续学习C的基础&#xff0c;那么废话不多说&#xff0c;我们正式进入今天的学习 C中的引用 1.1引用的概念 引用不是新定义一个变量&#xff0c;而是给已存在变量取了一个别名&#xff0…

uniapp小程序:大盒子包裹小盒子但是都有点击事件该如何区分?

在开发过程中我们会遇到这种情况&#xff0c;一个大盒子中包裹这一个小盒子&#xff0c;两个盒子都有点击事件&#xff0c;例如&#xff1a; 这个时候如果点击评价有可能会点击到它所在的大盒子&#xff0c;如果使用css中的z-index设置层级的话如果页面的盒子多的话会混乱&…

Spring解决泛型擦除的思路不错,现在它是我的了。

你好呀&#xff0c;我是浮生。 Spring 的事件监听机制&#xff0c;不知道你有没有用过&#xff0c;实际开发过程中用来进行代码解耦简直不要太爽。 但是我最近碰到了一个涉及到泛型的场景&#xff0c;常规套路下&#xff0c;在这个场景中使用该机制看起来会很傻&#xff0c;但…

15、FreeRTOS 软件定时器

文章目录 一、什么是定时器?1.1 定时器的理解1.2 软件定时器的特性 二、 软件定时器的上下文2.1 守护任务2.2 守护任务的调度2.3 回调函数 三、软件定时器的函数3.1 创建3.2 删除3.3 启动/停止3.5 修改周期3.6 定时器ID 四、案例4.1 一般使用4.2 消除抖动 一、什么是定时器? …