【c++】模拟实现优先级队列(priority_queue)

news2024/9/27 21:19:07

全部代码

  • 以容器适配器的玩法来实现,底层容器默认为vector

  • 使用了模板参数T表示存储在队列中的元素类型,Container表示底层容器类型,默认为vector,Compare表示比较器类型,默认为less。

  • adjustDown函数用于向下调整堆,保持堆的性质。它从指定的父节点开始,将其与子节点进行比较,如果子节点的值更大,则交换父节点和子节点的位置,并继续向下调整直到满足堆的性质。

  • adjustUp函数用于向上调整堆,保持堆的性质。它从指定的子节点开始,将其与父节点进行比较,如果子节点的值更大,则交换父节点和子节点的位置,并继续向上调整直到满足堆的性质。

  • 构造函数可以接受一个迭代器范围[first, last],用于初始化优先队列。在构造过程中,首先将迭代器范围内的元素存储到底层容器中,然后从最后一个非叶子节点开始,依次调用adjustDown函数,使得整个容器满足堆的性质。

  • empty函数用于判断优先队列是否为空,即底层容器是否为空。

  • size函数用于返回优先队列中元素的个数,即底层容器的大小。

  • top函数用于返回优先队列中的最大元素(根节点),但并不删除该元素。

  • push函数用于将一个元素插入到优先队列中。它将元素添加到底层容器的末尾,并调用adjustUp函数向上调整堆。

  • pop函数用于删除优先队列中的最大元素(根节点)。它首先将根节点与最后一个叶子节点交换位置,然后删除最后一个叶子节点,并调用adjustDown函数向下调整堆。

#pragma once
#include<vector>
namespace hqj
{
    template <class T, class Container = vector<T>, class Compare = less<T> >

    class priority_queue

    {
    private:
        void adjustDown(int parent)
        {
            int child = parent * 2 + 1;

            while (child < _c.size())
            {
                if (child + 1 < _c.size() && _comp(_c[child], _c[child+1]))
                {
                    child++;
                }

                if (_comp(_c[parent], _c[child]))
                {
                    swap(_c[parent], _c[child]);
                    parent = child;
                    child = parent * 2 + 1;
                }
                else
                {
                    break;
                }
             
            }
        }

        void adjustUp(int child)
        {
            int parent = (child - 1) / 2;
            while (child > 0)
            {
                if (_comp(_c[parent],_c[child]))
                {
                    swap(_c[parent], _c[child]);
                    child = parent;
                    parent = (child - 1) / 2;
                }
                else
                {
                    break;
                }
            }
        }
    public:

        priority_queue()
        {}

        template <class InputIterator>

        priority_queue(InputIterator first, InputIterator last)
            :_c(first,last)
        {
            for (int i = _c.size() - 1 - 1; i >= 0; i++)
            {
                adjustDown(i);
            }
        }

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

        size_t size() const
        {
            return _c.size();
        }

        T& top() const
        {
            return (T&)_c.front();
        }

        void push(const T& x)
        {
            _c.push_back(x);
            adjustUp(_c.size() - 1);
        }

        void pop()
        {
            swap(_c.front(), _c.back());
            _c.pop_back();
            adjustDown(0);
        }

    private:

        Container _c;

        Compare _comp;

    };

};

私有成员

  • _c容器对象:缺省的容器类型是vector

  • _comp比较器对象

Container _c;
Compare _comp;

构造函数

  • 由于我们模拟实现的优先级队列是一个容器适配器,私有成员中不含内置类型。利用系统生成的默认构造函数特性,会自动调用私有成员的构造函数,所以我们可以不写该优先级队列的构造函数的内容

 priority_queue()
      {}

析构函数

  • 同理根据系统默认生成的析构函数特性,当对象销毁时会自动调用自定义类型的析构函数,我们也可以不写

向下调整函数

  • 向下调整函数的参数是父节点的下标

  • 首先我们通过父亲下标找到其左孩子

  • 随后我们通过比较左右孩子大小来确定父亲要与哪个孩子进行比较、交换,至于是选择较大孩子还是较小孩子要根据比较器_comp的返回值来确定,如果我们想建小堆则选取较小的孩子;若我们要建大堆,则选取较大的孩子

  • 交换父亲和孩子,并且更新父亲和孩子

  • 由于向下调整次数不一定唯一,我们需要用到while结构,循环终止条件为:child下标越界、父子间的大小关系不满足比较器的要求。

 void adjustDown(int parent)
      {
          int child = parent * 2 + 1;

          while (child < _c.size())
          {
              if (child + 1 < _c.size() && _comp(_c[child], _c[child+1]))
              {
                  child++;
              }

              if (_comp(_c[parent], _c[child]))
              {
                  swap(_c[parent], _c[child]);
                  parent = child;
                  child = parent * 2 + 1;
              }
              else
              {
                  break;
              }
           
          }
      }

向上调整函数

  • 向上调整函数的参数是孩子的下标

  • 首先通过孩子的下标找到父亲的下标

  • 当父子关系满足比较器_comp的要求时进行调整,交换父亲和孩子,并更新父亲和孩子的下标

  • 同样,向上调整不止一次,需要用到while结构,循环终止条件为:孩子为根节点、父亲和孩子间的大小关系不满足比较器_comp要求

  void adjustUp(int child)
      {
          int parent = (child - 1) / 2;
          while (child > 0)
          {
              if (_comp(_c[parent],_c[child]))
              {
                  swap(_c[parent], _c[child]);
                  child = parent;
                  parent = (child - 1) / 2;
              }
              else
              {
                  break;
              }
          }
      }

迭代器构造函数

  • 由于模拟实现的优先级队列是容器适配器,直接使用底层容器的迭代器构造就行了,读入要建堆的数据

  • 都读入后,先找到最后一个叶子节点的父亲节点,随后传递该节点下标进向下调整函数,开始依次向下调整

  • 以arr数组为例:

   priority_queue(InputIterator first, InputIterator last)
          :_c(first,last)
      {
          for (int i = _c.size() - 1 - 1; i >= 0; i--)
          {
              adjustDown(i);
          }
      }

empty函数

  • 还是底层容器接口的复用,调用底层容器的empty函数

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

size函数

  • 底层容器接口的复用,调用底层容器的size函数

   size_t size() const
      {
          return _c.size();
      }

top函数

  • 函数作用是返回优先级队列队头元素(也就是堆顶元素),而该元素正好是底层容器的首元素,复用front接口就行,记得要强转

     T& top() const
     {
         return (T&)_c.front();
     }

push函数的实现

  • 调用底层容器的push_back函数实现插入功能,然后再进行向上调整堆

      void push(const T& x)
      {
          _c.push_back(x);
          adjustUp(_c.size() - 1);
      }

pop函数的实现

  • 首先交换队头元素和队尾元素(堆顶元素和堆尾元素)

  • 将队尾元素删除

  • 从堆顶开始向下调整堆

      void pop()
      {
          swap(_c.front(), _c.back());
          _c.pop_back();
          adjustDown(0);
      }

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

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

相关文章

高速下载b站视频的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

【C语言】关于char的取值范围的讨论

前提知识&#xff1a; 计算机内存中存的是整数的补码。 正数的原反补相同&#xff01; 负数的补码 &#xff08;除符号位以外&#xff09;原码取反 1 负数的源码 &#xff08;除符号位以外&#xff09;补码取反 1 有符号的char&#xff0c;最高位二进制位表示符号位 …

基于 MTAOO 方法论,看连锁餐饮品牌如何落地 CJO 理念、实现精细化用户运营

移动互联网时代的流量红利已经消失&#xff0c;企业亟需在数字化转型时代抓住触点红利&#xff0c;基于客户旅程编排&#xff08;Customer Journey Orchestration&#xff0c;简称 CJO&#xff09;为用户提供个性化、全渠道一致的体验。 在此背景下&#xff0c;连锁餐饮品牌已经…

《C和指针》笔记34:字符串函数

文章目录 1. 获取字符串长度strlen 2. 复制字符串strcpystrncpy 3. 拼接字符串strcatstrncat 4. 字符串比较strcmpstrncmp 1. 获取字符串长度 strlen 库函数strlen的原型如下&#xff1a; size_t strlen( char const *string );注意strlen返回一个类型为size_t的值。这个类型…

【GIT】:一文快速了解什么是GIT

【GIT】&#xff1a;一文快速了解什么是GIT 个人主页: 【⭐️个人主页】 需要您的【&#x1f496; 点赞关注】支持 &#x1f4af; 关于版本控制 什么是“版本控制”&#xff1f;我为什么要关心它呢&#xff1f; 版本控制是一种记录一个或若干文件内容变化&#xff0c;以便将来…

使用数组实现队列

目录 队列的应用场景 任务调度 广度优先搜索&#xff08;BFS&#xff09; 网络请求管理 消息队列 当我们在编写JavaScript代码时&#xff0c;经常会遇到需要使用队列的情况。队列是一种常见的数据结构&#xff0c;它按照先进先出&#xff08;First-In-First-Out&#xff0…

【刷题篇】笔试真题

文章目录 复数乘法一年中的第几天字符串相加字符串相乘 复数乘法 复数 可以用字符串表示&#xff0c;遵循 “实部虚部i” 的形式&#xff0c;并满足下述条件&#xff1a; 实部 是一个整数&#xff0c;取值范围是 [-100, 100] 虚部 也是一个整数&#xff0c;取值范围是 [-100, 1…

ES6初步了解迭代器

迭代器是什么&#xff1f; 迭代器(iterator)是一种接口&#xff0c;为各种不同的数据结构提供统一的访问机制。任何数据结构只要部署 iterator 接口&#xff0c;就可以完成遍历操作 ES6创造了一种新的遍历方法for…of循环&#xff0c;iterator 接口主要供 for…of 使用 原生中具…

docker、docker-compose安装教程,很详细

docker、docker-compose安装教程&#xff0c;很详细 一、卸载旧版1、查看有没有安装过旧版2、停止docker3、删除安装过docker的相关包4、删除docker相关的镜像和容器 二、docker安装1、设置阿里云镜像2、查看所有docker3、安装最新版本4、安装指定版本 三、使用前准备1、启动do…

第二证券:AIGC概念活跃,焦点科技、三维通信涨停,万兴科技大涨

AIGC概念24日盘中走势生动&#xff0c;到发稿&#xff0c;万兴科技、三态股份涨超10%&#xff0c;焦点科技、三维通讯、我国科传等涨停&#xff0c;中文在线涨超9%&#xff0c;果麦文明、新国都涨约7%。 消息面上&#xff0c;各大电商途径于10月18-24日先后发动“双11”大促或…

FPGA驱动步进电机-Sin曲线加速

FPGA驱动步进电机-Sin曲线加速 基本实现原理实际仿真的波形程序 以下由特权同学的FPGA文档摘取 Sin 曲线控制 step 脉冲信号生成的功能框图如下所示。 基本实现原理 ①判断步进电机驱动的目标频率 stepper_delay_target 与当前频率 stepper_delay_current的值是否一致&#…

Java IDEA controller导出CSV,excel

Java IDEA controller导出CSV&#xff0c;excel 导出excel/csv&#xff0c;亲测可共用一个方法&#xff0c;代码逻辑里判断设置不同的表头及contentType&#xff1b;导出excel导出csv 优化&#xff1a;有数据时才可以导出参考 导出excel/csv&#xff0c;亲测可共用一个方法&…

【Jenkins 安装】

一&#xff1a;安装文件夹准备 在/home/admin 界面下新建三个文件夹&#xff0c;用来安装tomcat、maven 1.打开&#xff0c;/home/admin目录 cd /home/admin 2.新建三个文件夹 mkdir tomcat mkdir maven 二&#xff1a;安装tomcat 1.打开tomcat目录进行tomcat的安装 访问:h…

Xfigure综合膳食营养粉美丽上线,大健康行业竞争呈现多元化

10月21日&#xff0c;“美丽健康 营养为先”2023全民营养健康科学论坛暨悦小妖2023秋季新品发布会在杭州召开&#xff0c;会上就当下的国民营养健康问题提出了许多建设性的观点&#xff0c;新发布的Xfigure是行业内少有的提倡营养为主的特膳类产品。 拥抱趋势&#xff0c;全新突…

redis持久化之RDB(Redis DataBase)

1 : 总体介绍 Redis是一个基于内存的数据库&#xff0c;它的数据是存放在内存中&#xff0c;内存有个问题就是关闭服务或者断电会丢 失。 Redis的数据也支持写到硬盘中&#xff0c;这个过程就叫做持久化 1.1 。 Redis提供了2种不同形式的持久化方式。 RDB&#xff08;Redis Da…

uboot移植之DDR初始化参数更改说明

一. 简介 裸机篇开发时&#xff0c;DDR初始化是 imxdownload软件完成的。imxdownload软件在 二进制文件 u-boot.bin 前面加上头部(IVT、DCD等数据)。这其中所加的头部信息就包括 DDR初始化内容。 u-boot.bin 就是编译出来的 uboot 二进制文件。 uboot 是个裸机程序&#x…

JUnit5参数化测试的几种方式!

参数化测试一直是津津乐道的话题&#xff0c;我们都知道JMeter有四种参数化方式&#xff1a;用户自定义变量、用户参数、CSV文件、函数助手&#xff0c;那么JUnit5有哪些参数化测试的方式呢&#xff1f; 依赖 JUnit5需要添加junit-jupiter-params依赖才能使用参数化&#xff…

Java项目_家庭记账(简易版)

文章目录 简介代码实现 简介 该项目主要用来练习&#xff0c;Java的变量&#xff0c;运算符&#xff0c;分支结构和循环结构的知识点。 程序界面如下&#xff1a; 登记收入 登记支出 收支明细 程序退出 代码实现 package project;import java.util.Scanner;import sta…

【疑问解决】- 源码Enmu枚举类的toString里面的name是哪里来的,什么时候传入的?

起因是听课到 该段的输出boy输出什么&#xff1f; 答案就是输出BOY&#xff0c;但韩老师解释的有点笼统。 但是我看了一眼源码关于这个name确实有点没头绪 public abstract class Enum<E extends Enum<E>>implements Comparable<E>, Serializable {/*** T…

可自由搭建的能源管理平台,轻松实现高效节能

随着科技的不断发展&#xff0c;能源问题越来越重要。为了提高能源的利用效率&#xff0c;减少能源浪费&#xff0c;能源用能企业纷纷开始注重能源管理工作&#xff0c;并想要一款可以进行高效管理的工具。智慧能源管理平台&#xff0c;是一款可自由搭建的能源管理平台&#xf…