Priority_Queue 的使用和模拟

news2025/1/9 12:49:03
目录
一·基本的介绍

优先队列是一种容器适配器;他的第一个元素总是他包含所有元素里面最大的一个

他的底层容器可以是任何标准容器类模板,也可以是其他特定设计的容器类。
这个底层容器应该可以通过随机访问迭 代器,并支持以下操作:
empty():检测容器是否为空
size():返回容器中有效元素个数
front():返回容器中第一个元素的引用
push_back():在容器尾部插入元素
pop_back():删除容器尾部元素

优先队列里面的“优先”指的是:进行容器遍历访问 的时候优先访问顶部的元素

二. 常用接口的介绍 
1.构造函数

2. void push(const T&val ) 

3. void pop()

 4. const T& top( )

 

 接口的简单使用:

 通过对成员接口的了解,我们可以推测优先队列的底层结构可能是一个

三· 模拟实现
1.仿函数

第一个参数用到的就是仿函数。

仿函数是对 opertaor() 的一个重载

2.仿函数使用
2.1  指定进行降序的输出
2.2 指定进行升序的输出 

2.3 仿函数的调用 

分析:

Less <Date> l2,编译器进行实例化的时候,模板参数T 实例化为 Date 类型,从而生成一份

bool operator() (Date left,Date right)的函数,只不过使用模板,咱们程序员省了此步骤。 

3.priority_queue 类的模拟

一般默认Container 的类型是  vector<T>

为什么默认使用 vector 作为优先队列的底层容器?而不用list这个容器

1) 优先队列支持快速的访问此队列里面的最大或者最小元素;因此底层通常借助堆这一结构实

现,堆是一个特殊的完全二叉树

2)vector 提供了动态大小数组的功能,内部是一段连续的内存,因此在访问元素的时候,大大提

高了效率

3)list 相比较vector 而言,内存是不连续的,在进行元素访问的时候,需要对指针进行一系列操

作,因此效率不如vector

4 push() 模拟

因为需要随时保证访问的第一个元素是最大的或是最小的,这里需模拟建大堆和建小堆的一系列操

有关对堆进行上调和下调的相关细节的实现,可以康康下面博客

 建堆的相关调整

4.1 上调算法
  void adjust_up(int child) //从孩子节点开始
        {
            Compare com;
            int parent = (child - 1 ) / 2;//默认根节点下标从0开始
            while (child > 0)
            {
                if (_c[parent] < _c[child]) //对象直接比较
                //if(com(_c[parent] , _c[child]))  // _com(_c[parent] , _c[child] 相当于回调仿函数模板,可以把_com 想象成函数指针,仿函数模板:替代函数指针
                {
                    std::swap(_c[parent], _c[child]);
                    child = parent;
                    parent = (child - 1) / 2;
                }
                else
                    break;
            }
        }
4.2 push ()实现
  void push(const T& x)
        {
            _c.push_back(x);
            // 上调:建大堆
            adjust_up(_c.size()-1);//需要把插入当前数据所在的位置传过去
        }
5.pop()模拟
5.1 下调算法
        void adjust_down(int parent) //向下调整从父节点开始
        {
            Compare com;
            int child = 2 * parent + 1;
            while ((size_t)child < (_c.size() ))
            {
                if ((size_t)child + 1 < (_c.size() ) 
                    && _c[child + 1] > _c[child]) // _c[child + 1] > _c[child] 这里直接就是对象的大小比较,可以使用仿函数进行大小比较
              

                    child++;//更新为最大的孩子节点
                //if (_c[child] > _c[parent] )
                if (com( _c[parent], _c[child]) ) //使用仿函数模板
                {
                    std::swap(_c[child], _c[parent]);
                    parent = child;
                    child = 2 * parent + 1;
                }
                else

                    break;
                
            }
       }
5.2 pop()实现
        void pop() 
        {
            //堆的删除:堆顶与堆尾交换在进行大堆的调整
            std::swap(_c[0], _c[_c.size() - 1]);
            _c.pop_back();

            adjust_down(0);
        }
6. top()模拟
        T& top()
        {
            return _c[0];
        }
7.size()

8. empty()

9. 完整代码实现 
#pragma once
#include<vector>
namespace y
{
    //仿函数模板
    //就是对 () 进行重载
   
	template<class T, class Container = vector<T>,class Compare = Less<T>>  //注意:Compare 只是一个模板类型,实例化的时候,会变成指定类型
	class priority_queue
	{
    public:

        priority_queue()
        {}

        template <class InputIterator>

        priority_queue(InputIterator first, InputIterator last)
            :_c(first,last)
        {
            //建议使用下调时间复杂度 O(N)
            //必须是有序,所有一开始传i = (_c.size() - 1 -1)/2 对应最后一个父节点
            for (int i = (_c.size() - 1 -1)/2; i >= 0; --i)
            {
                adjust_down(i);
            }
        }

        bool empty() const
        {
            return _c.empty();
        }

        size_t size() const
        {
            return _c.size();
        }
        void adjust_down(int parent) //向下调整从父节点开始
        {
            Compare com;
            int child = 2 * parent + 1;
            while ((size_t)child < (_c.size() ))
            {
                if ((size_t)child + 1 < (_c.size() ) 
                    //&& _c[child + 1] > _c[child]) // _c[child + 1] > _c[child] 这里直接就是对象的大小比较,可以使用仿函数进行大小比较
                    && com(_c[child], _c[child+1]))// 建大堆

                    child++;//更新为最大的孩子节点
                //if (_c[child] > _c[parent] )
                if (com( _c[parent], _c[child]) ) //使用仿函数模板
                {
                    std::swap(_c[child], _c[parent]);
                    parent = child;
                    child = 2 * parent + 1;
                }
                else

                    break;
                
            }
       }
        void adjust_up(int child) //从孩子节点开始
        {
            Compare com;
            int parent = (child - 1 ) / 2;//默认根节点下标从0开始
            while (child > 0)
            {
                //if (_c[parent] < _c[child]) //对象直接比较
                if(com(_c[parent] , _c[child]))  // _com(_c[parent] , _c[child] 相当于回调仿函数模板,可以把_com 想象成函数指针,仿函数模板:替代函数指针
                {
                    std::swap(_c[parent], _c[child]);
                    child = parent;
                    parent = (child - 1) / 2;
                }
                else
                    break;
            }
        }
        void pop() 
        {
            //堆的删除:堆顶与堆尾交换在进行大堆的调整
            std::swap(_c[0], _c[_c.size() - 1]);
            _c.pop_back();

            adjust_down(0);
        }

        void push(const T& x)
        {
            _c.push_back(x);
            // 上调:建大堆
            adjust_up(_c.size()-1);//需要把插入当前数据所在的位置传过去
        }

        T& top()
        {
            return _c[0];
        }

    private:

        Container _c;//表示底层的容器

        //Compare _com;//类似于函数指针
	};
}

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

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

相关文章

【LINUX】ifconfig -a查看到的发送、接收包数和字数字节数在驱动层代码大概位置

先看结果 ifconfig -a查看到发送的信息&#xff1a; 从上图可以看出来网卡驱动代码的目录是在drivers/net/ethernet/intel/e1000/e1000_main.c 下图是接收到的信息&#xff1a; 不过这些数据是在虚拟机看到的&#xff0c;如果有条件可以在实际的物理网卡测试看看效果。 下边这…

IO进程线程 0828作业

作业 有名管道&#xff0c;创建两个发送接收端&#xff0c;父进程写入管道1和管道2&#xff0c;子进程读取管道2和管道1 mkfifo1.c代码 #include <myhead.h> int main(int argc, const char *argv[]) {if(mkfifo("./my_fifo1",0664) -1){perror("mkfi…

轻量级数据库

在计算机编程中&#xff0c;句柄&#xff08;Handle&#xff09;是一种用于标识和引用系统内部对象的标识符。句柄通常是一个不透明的指针或整数值&#xff0c;它代表了一个系统资源&#xff0c;如文件、窗口、进程、线程、内存映射文件等。句柄的主要作用是在程序中引用这些资…

【Docker】搭建docker的私有仓库

一、为什么搭建私有仓库 docker hub虽然方便&#xff0c;但是还是有限制 需要internet连接&#xff0c;速度慢所有人都可以访问由于安全原因企业不允许将镜像放到外网 好消息是docker公司已经将registry开源&#xff0c;我们可以快速构建企业私有仓库 二、搭建简单的Registr…

《深入浅出WPF》读书笔记.10资源

《深入浅出WPF》读书笔记.10资源 背景 这一章主要讲资源&#xff0c;包括StaticResource&#xff0c;DynamicResource&#xff0c;以及二进制资源等等 资源 ResourceDictionary 资源对象为object类型&#xff0c;需要自己进行类型转换 <Window x:Class"ResourceDe…

【产品那些事】什么是软件成分分析(SCA)?

文章目录 前言背景SCA概述 功能组成工作原理软件供应链漏洞检测通用架构组件的数量和检测算法方面基于包管理器(构建)检测技术有那些检测算法&#xff1f;引用开源软件的方式方面漏洞库的积累与颗粒度 SCA产品Depency-Check基本介绍工作原理文件类型分析器工具使用体验Maven插件…

CS2饰品价格趋势怎么看?以及最佳入手时机

CS2饰品价格趋势怎么看?以及最佳入手时机 CS2饰品价格趋势怎么看?以及最佳入手时机 CS2选品时价格趋势图到底怎么看&#xff1f;什么时候值得真正入手&#xff1f;&#xff1f; 8月中上旬这波涨势大家抓住了吗&#xff1f;反正我们是抓住了。然而很多人都是听别人说行情上涨…

squareTest 破解

下载jclasslib 使用brew下载 brew install jclasslib-bytecode-viewer 准备工作 1. IDEA插件squareTest安装 2. 字节码查看器jclasslib下载&#xff1a;https://github.com/ingokegel/jclasslib/releases 破解流程(mac为例) 1. 首先找到插件jar包所在位置&#xff0c;mac…

RocketMQ第5集

一 RocketMQ的工作流程 1.1 生产环节producer Producer可以将消息写入到某Broker中的某Queue中&#xff1a;其中Producer发送消息之前&#xff0c;会先向NameServer发出获取消息Topic的路由信息的请求&#xff0c;NameServer返回该Topic的路由表及Broker列表。简单的说&…

【CPP 基础】如何把cpp库,分装给 c# 用。

基本方法 在C中封装的方法&#xff08;如在DLL中&#xff09;&#xff0c;如果输入参数是一个自定义类型&#xff0c;并且你想在C#中调用它&#xff0c;你需要做一些工作来确保数据结构在两种语言之间正确传递和解释。下面是详细的步骤。 场景描述 假设我们有一个C函数&#…

openjudge.4.6算法之贪心_746:Elevator Stopping Plan

题目 746:Elevator Stopping Plan 总时间限制: 1000ms 内存限制: 65536kB 描述 ZSoft Corp. is a software company in GaoKe Hall. And the workers in the hall are very hard-working. But the elevator in that hall always drives them crazy. Why? Because there is on…

计算机常见运算之左移操作、右移操作以及按位与、按位或

文章目录 前言一、左移操作&#xff08;<<&#xff09;和 右移操作&#xff08;>>&#xff09;1.1 左移操作&#xff08;<<&#xff09;1.2 右移操作&#xff08;>>&#xff09;1.3 应用场景 二、按位与 (&) 和 按位或 (|)2.1 按位与 (&)2.2 按…

Java、python、php版 剧本杀拼团服务平台 剧本杀管理系统(源码、调试、LW、开题、PPT)

&#x1f495;&#x1f495;作者&#xff1a;计算机源码社 &#x1f495;&#x1f495;个人简介&#xff1a;本人 八年开发经验&#xff0c;擅长Java、Python、PHP、.NET、Node.js、Android、微信小程序、爬虫、大数据、机器学习等&#xff0c;大家有这一块的问题可以一起交流&…

谓词和量词

一、个体词和谓词 命题是一句陈述句&#xff0c;命题由个体词和谓词组成。 个体词是句子中的主语部分&#xff0c;比如这里的王童。 谓词是句子里的剩余部分&#xff0c;比如是一个三好学生 个体词用小写字母表示&#xff0c;谓词用大写字母&#xff0b;&#xff08;&#…

网络编程(学习)2024.8.29

目录 阻塞式IO(BIO) 特点 阻塞原因与阻塞反应 TCP流式套接字缓冲区 非阻塞式IO(NIO) 特点 设置非阻塞 1.通过对参数的修改实现 2.通过对文件描述符的属性进行设置 fcntl 信号驱动IO (异步IO模型) IO多路复用 select、poll、epoll IO多路复用机制 1.select …

深度学习实战2--MNIST 手写数字分类(代码在末尾)

1.本节目标&#xff1a; (1)了解什么是MNIST 数据集&#xff1b; (2)了解卷积神经网络对图片处理的流程&#xff1b; (3)能够看懂Python 编写的对图片分类任务的代码&#xff1b; (4)在一定程度上掌握处理类似任务的编程能力。 注意&#xff1a;本章节使用开源机器学习库P…

docker 部署 kkFileView 并 使用Nginx代理

拉取镜像 docker pull keking/kkfileview 运行容器 docker run -it -d -p 8012:8012 keking/kkfileview --restart always 配置nginx location /preview {# 本地运行的kkFileView的地址proxy_pass http://127.0.0.1:8012;proxy_set_header Host $host;proxy_set_header X-…

Sang.UAParser一个简单的.NET用户代理解析器

本文主要介绍了 Sang.UAParser 这个简单的.NET用户代理解析器&#xff0c;可以用来解析用户代理字符串&#xff0c;提取出其中的浏览器、操作系统等信息。这个库的使用非常简单&#xff0c;只需要引用 NuGet 包&#xff0c;然后调用相应的方法即可。 1. 简介 Sang.UAParser 是…

YOLO | YOLO目标检测算法(基础入门)

github&#xff1a;https://github.com/MichaelBeechan CSDN&#xff1a;https://blog.csdn.net/u011344545 YOLO目标检测算法 深度学习经典检测方法1、两阶段&#xff08;Two-stage&#xff09;2、单阶段&#xff08;One-stage&#xff09; 深度学习经典检测方法 1、两阶段&a…

blender修改材质时出现颜色丢失的问题

对于建立的三维模型&#xff0c;我们一般是直接使用gazebo时不会有材质的颜色信息&#xff0c;这一点还是比较烦的&#xff0c;所以这里通过blender来重新给模型上色 首先需要去安装blender&#xff1a; sudo apt install blender对于导入的模型,修改了材质后依然表现为没有颜…