『C++ - STL』之优先级队列( priority_queue )

简介: 『C++ - STL』之优先级队列( priority_queue )

前言

什么是优先级队列,从该名中可以知道他一定有队列的一定属性,即先入先出(LILO),而这里的优先级则可以判断出它的另一个特点就是可以按照一定的条件将符合该条件的先进行出队,这就是优先级队列;

而在数据结构中有一个支持该操作的结构 - 堆( heap );

而在STL中,这个优先级队列( priority_queue )也正是堆;


优先级队列的结构

既然优先级队列的结构是堆,那想必结构上也不难;

堆的结构是以顺序表为基础,从而实现完全二叉树的结构;

从该容器的接函数接口中也可以知道实际上它就是个堆;


优先级队列的模拟实现

优先级队列priority_queue为一个类模板容器;

且同STL中的栈stack与队列queue一样都为适配器模式的容器,即以某个容器为基础;

template <class T, class Container = vector<T>,class Compare = less<typename Container::value_type> > class priority_queue;

其模板参数有三个分别为:

  • class T
    容器所存储的数据类型 T ;
  • class Container = vector<T>
    容器适配器且定缺省参数默认为 vector< T >;
  • class Compare = less<typename Container::value_type>
    一个用来比较大小的仿函数,给定缺省参数默认为 less ,该仿函数在标准库std中;

根据文档中的信息来看,优先级队列主要的几个接口也正是数据结构中堆应有的结构;

#pragma once 
#include<iostream>
#include<vector>
#include<assert.h>
using namespace std;
namespace my_priority{//命名空间
  template<class T,class Container = std::vector<T>>//暂未设置仿函数
    class priority_queue{//总体框架
 public:
        void push(const T& val){//增
          _con.push_back(val);
          adjust_up(_con.size()-1);
        }
        void pop(){
          std::swap( _con[_con.size()-1],_con[0]);//删
          _con.pop_back();
          adjust_down(0);
        }
        bool empty(){//判空
          return _con.empty();
        }
        size_t size(){//返回大小
          return _con.size();
        }
        const T& top()const{//返回堆顶
          assert(!_con.empty());
          return _con[0];
 }
        void swap(priority_queue& con){//交换
          if(_con!=con._con)
          _con.swap(con._con);
        }
      private:
  //必要函数 - 向上调整&&向下调整
        void adjust_up(size_t child){
          size_t parent = child;
          while(parent>0){
            parent = (child-1)/2;
            if(_con[parent]<_con[child]){
              std::swap(_con[parent],_con[child]);
            }
            child = parent;
          }
        }
        void adjust_down(size_t parent){
          Compare comfunc;
          size_t child = parent*2+1;
          while(child<_con.size()){
            if(child+1<_con.size()&&_con[child]<_con[child+1]){
              ++child;
            }if(_con[parent]<_con[child]){
              std::swap(_con[parent],_con[child]);
            }
            parent = child;
            child = parent*2+1;
          }
        }
        Container _con; //容器适配器所实例化的对象,当前代码为vector<int>
    };

但是在库中,模板参数共有三个,具体的第三个仿函数到底是什么?


仿函数

仿函数,也被称为函数对象;

即一个可以使用函数功能的类,本质上就是在类中重载了operator();

举个简单的例子,当我们想要写一个能将两个数进行相加的仿函数即可以这么写;

struct Add{
  int operator()(int a,int b){
    return a+b;
  } 
}
int main()
{
  Add addfunc;
  int ret = addfunc(1,2);//仿函数的调用;
  ret = Add() (1,2);//利用匿名对象;
  return 0;
}

同理,也可以根据该方法写一个比较两个对象大小的仿函数;

为了能接受多种类型的数据进行比较也可以将其设置为类模板;

template<class T>
struct less{
  bool operator()(const T& a,const T& b){
    return a<b;
  }

这也就是在实现当中缺失的模板参数,仿函数;

由于优先级队列priority_queue的大小根堆属性是由其中的向上调整算法adjust_up与向下调整算法adjust_down来决定的(建堆以及堆的调整);所以只要将对应的比较大小><换成仿函数即可;


最终代码

#pragma once 
#include<iostream>
#include<vector>
#include<assert.h>
using namespace std;
namespace my_priority{
  template<class T>
    struct less{
      bool operator()(const T& v1,const T& v2){
        return v1<v2;
      }
    };
  template<class T>
    struct greater{
      bool operator()(const T& v1,const T& v2){
        return v1>v2;
      }
    };
  template<class T,class Container = std::vector<T> ,class Compare = less<T>>
    class priority_queue{
 public:
        void push(const T& val){
          _con.push_back(val);
          adjust_up(_con.size()-1);
        }
        void pop(){
          std::swap( _con[_con.size()-1],_con[0]);
          _con.pop_back();
          adjust_down(0);
        }
        bool empty(){
          return _con.empty();
        }
        size_t size(){
          return _con.size();
        }
        const T& top()const{
          assert(!_con.empty());
          return _con[0];
 }
        void swap(priority_queue& con){
          if(_con!=con._con)
          _con.swap(con._con);
        }
      private:
        void adjust_up(size_t child){
          Compare comfunc;
          size_t parent = child;
          while(parent>0){
            parent = (child-1)/2;
            if(comfunc(_con[parent],_con[child])){
              std::swap(_con[parent],_con[child]);
            }
            child = parent;
          }
        }
        void adjust_down(size_t parent){
          Compare comfunc;
          size_t child = parent*2+1;
          while(child<_con.size()){
            if(child+1<_con.size()&&comfunc(_con[child],_con[child+1])){
              ++child;
            }if(comfunc(_con[parent],_con[child])){
              std::swap(_con[parent],_con[child]);
            }
            parent = child;
            child = parent*2+1;
          }
        }
        Container _con;
    };
相关文章
|
1月前
|
存储 程序员 C++
C++常用基础知识—STL库(2)
C++常用基础知识—STL库(2)
71 5
|
1月前
|
存储 算法 调度
【C++打怪之路Lv11】-- stack、queue和优先级队列
【C++打怪之路Lv11】-- stack、queue和优先级队列
34 1
|
1月前
|
存储 自然语言处理 程序员
C++常用基础知识—STL库(1)
C++常用基础知识—STL库(1)
58 1
|
1月前
|
算法 安全 Linux
【C++STL简介】——我与C++的不解之缘(八)
【C++STL简介】——我与C++的不解之缘(八)
|
1月前
|
算法 数据处理 C++
c++ STL划分算法;partition()、partition_copy()、stable_partition()、partition_point()详解
这些算法是C++ STL中处理和组织数据的强大工具,能够高效地实现复杂的数据处理逻辑。理解它们的差异和应用场景,将有助于编写更加高效和清晰的C++代码。
25 0
|
2天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
15 2
|
8天前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
33 5
|
14天前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
46 4
|
15天前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
43 4
|
1月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
28 4