• C++学习——优先级队列模拟实现与仿函数初步认识


    目录

    ​编辑

    一,优先级队列

    二,实现

    1.构造priority_queue类

    2.简单的top()与size()还有empty()函数

    3.push函数

    4.pop函数

    5.构造函数

    6.测试

    三,仿函数

    1.介绍

     2.使用


    一,优先级队列

    优先级队列——priority_queue。这个结构体类型也是C++里面的一个非常好用的一个数据结构。可以在Cplusplus网站上看看这个数据结构的介绍:

    在这一项介绍中可以看到这个数据结构类型的各种模板。因为我们要实现一个priority_queue所以我们就要来看看这个数据结构类型的各种函数,如下:

    今天我们只会实现size,top,push,pop,还有一个construct的迭代器构造函数。在这里还要声明一点,这个数据结构类型的地城原理其实就是一个堆,也就是heap。

    二,实现

    1.构造priority_queue类

    实现步骤如下,首先开一个.h文件。然后写下如下代码:

    1. #include
    2. 2 using namespace std;
    3. 3
    4. 4 namespace area
    5. 5 {
    6. 6 template<class T,class Container>
    7. 7 class priority_queue
    8. 8 {
    9. 9 public:
    10. 10
    11. 11
    12. 12 private:
    13. 13 Container _con;
    14. 14
    15. 15 };
    16. 16
    17. 17 }

    因为这个优先级队列实现的原理和堆实现的原理相像,所以可以先传一个缺省参数vector给Container。如下:

    1. #include
    2. 2 using namespace std;
    3. 3 #include
    4. 4
    5. 5 namespace area
    6. 6 {
    7. 7 template<class T,class Container=vector >
    8. 8 class priority_queue
    9. 9 {
    10. 10 public:
    11. 11
    12. 12
    13. 13 private:
    14. 14 Container _con;
    15. 15
    16. 16 };
    17. 17
    18. 18 }

    2.简单的top()与size()还有empty()函数

    这两个函数实现起来比较简单,所以可以先来实现一下,实现代码如下:

    1. #include
    2. 2 using namespace std;
    3. 3 #include
    4. 4
    5. 5 namespace area
    6. 6 {
    7. 7 template<class T,class Container=vector >
    8. 8 class priority_queue
    9. 9 {
    10. 10 public:
    11. 11 const T& top()//top函数
    12. 12 {
    13. 13 return _con[0];
    14. 14 }
    15. 15
    16. 16 size_t size()//size()函数
    17. 17 {
    18. 18 return _con.size();
    19. 19 }
    20. 20
    21. 21 bool empty()//empty()函数
    22. 22 {
    23. 23 return _con.size()==0;
    24. 24 }
    25. 25 private:
    26. 26 Container _con;
    27. 27
    28. 28 };
    29. 29
    30. 30 }
    31. ~

    在这里复习一个知识点,top函数能使用引用传参的条件是_con[0]出了top函数的作用域以后没有被销毁。

    3.push函数

    因为我们要模仿的是堆排序的模式,所以在实现push函数的同时就得实现一个向上调整建堆。向上调整建堆的代码如下:

    1. void adjust_up(int child)
    2. 27 {
    3. 28 size_t parent = (child-1)/2;
    4. 29 while(child>0)
    5. 30 {
    6. 31 if(_con[child]<_con[parent])
    7. 32 {
    8. 33 swap(_con[child],_con[parent]);//模板数据结构里面已经实现这个swap()函数。
    9. 34 child = parent;
    10. 35 parent = (child-1)/2;
    11. 36 }
    12. 37 else
    13. 38 {
    14. 39 break;
    15. 40 }
    16. 41 }
    17. 42 }

    然后在push调用的时候便是这样使用的:

    1. void push(T& x)
    2. 45 {
    3. 46 _con.push_back(x);
    4. 47 adjust_up(_con.size()-1);
    5. 48 }

    当然,这段代码不能让这个堆的每个下标的大小都是符合顺序的但是可以保证这个vector类型的数组是一个堆。

    4.pop函数

    pop函数的实现也要依靠一个向下调整的函数。这个向下调整的函数实现代码如下:

    1. void adjust_down(int parent)
    2. 51 {
    3. 52 int child = parent*2+1;
    4. 53 while(child<_con.size())
    5. 54 {
    6. 55 if(child+1<_con.size()&&_con[child+1]<_con[child])
    7. 56 {
    8. 57 child++;
    9. 58 }
    10. 59 if(_con[child]<_con[parent])
    11. 60 {
    12. 61 swap(_con[child],_con[parent]);
    13. 62 parent = child;
    14. 63 child = parent*2+1;
    15. 64 }
    16. 65 else
    17. 66 {
    18. 67 break;
    19. 68 }
    20. 69
    21. 70 }
    22. 71 }

    然后我们再来实现pop()函数:

    1. void pop()
    2. 74 {
    3. 75 swap(_con[0],_con[_con.size()-1]);
    4. 76 _con.pop_back();
    5. 77 adjust_down(0);
    6. 78 }

    经过上述操作,我们的pop函数就写好了。

    5.构造函数

    1.迭代器构造函数:

    1. template<class InputIterator>
    2. 36 priority_queue(InputIterator first,InputIterator last)
    3. 37 :_con(first,last) //调用模板容器的迭代器构造函数
    4. 38 {
    5. 39 for(int i=(_con.size()-1-1)/2;i>=0;i--)
    6. 40 {
    7. 41 adjust_down(i); //从最后一个父节点开始调整成堆
    8. 42 }
    9. 43 }

    这个迭代器区间构造函数其实是十分好写的,因为我们调用的容器里面已经实现了迭代器区间构造函数,我们复用便可以了。但是这里要注意一个点,就是在我们写下这个迭代器区间构造函数以后,默认生成的无参的构造函数就没有了。所以我们就得加一个无参的构造函数:

    1. priority_queue()
    2. 31 {
    3. 32
    4. 33 }

    6.测试

    我们再创建一个.cc文件,写入以下代码:

    1. #include"priority_queue.h"
    2. 2
    3. 3 int main()
    4. 4 {
    5. 5 area::priority_queue<int>q;
    6. 6 q.push(4);
    7. 7 q.push(3);
    8. 8 q.push(9);
    9. 9 q.push(10);
    10. 10 q.push(40);
    11. 11 q.push(99);
    12. 12 q.push(12);
    13. 13 q.push(9);
    14. 14
    15. 15 while(!q.empty())
    16. 16 {
    17. 17 cout<top()<<" ";
    18. 18 q.pop();
    19. 19 }
    20. 20
    21. 21 cout<
    22. 22 return 0;
    23. 23 }

    运行结果如下:

    得到这样的结果说明我们的优先级队列是写对了的。

    三,仿函数

    1.介绍

    在上面的代码中,我们实现的是一个小堆但是如果我们要实现一个大堆呢?难道我们要改代码吗?我们当然是不用的,可以仿照qsort函数的实现手法:

    在这个函数实现的时候会有一个compar函数,这个函数是函数指针类型的我们可以通过这个写这个函数的逻辑来实现按顺序排还是按逆序排。在实现优先级队列时我们也可以根据这个思想来实现相同的控制效果。但是在Cpp中是非常鄙视函数指针的。所以我们得用到一个叫做仿函数的知识。仿函数如下:

    1. template<class T>
    2. 5 class Less
    3. 6 {
    4. 7 bool operator()(const T&x,const T&y)
    5. 8 {
    6. 9 return x
    7. 10 }
    8. 11 };
    9. 12
    10. 13 template<class T>
    11. 14 class Greater
    12. 15 {
    13. 16 bool operator()(const T&x,const T&y)
    14. 17 {
    15. 18 return x>y;
    16. 19 }
    17. 20 };

    这样便实现了两个仿函数了。可以看到仿函数其实不是一个函数,只不过是在一个类里面将

    ()重载,从而让这个类可以像函数一样使用,所以才叫做仿函数。

     2.使用

    实现了以上两个仿函数以后,我们便要开始在我们模拟实现的优先级队列里使用这个仿函数了。首先我们得在模板里面再添加一个模板参数compare,如下:

     template<class T,class Container=vector,class compare = Less >   

    并且默认生成的是小堆。然后我们再来一些替换,将比较的地方换成仿函数的对象。所以我们要先造一个仿函数对象:

     compare com;
    

    然后将比较的地方替换成仿函数便可以了,如以下地方:

    1. if(child+1<_con.size()&&com(_con[child+1],_con[child]))
    2. 78 {
    3. 79 child++;
    4. 80 }
    5. 81 if(com(_con[child],_con[parent]))
    6. 82 {
    7. 83 swap(_con[child],_con[parent]);
    8. 84 parent = child;
    9. 85 child = parent*2+1;
    10. 86 }

  • 相关阅读:
    CSS的概念和基本用法
    vue3 快速入门系列 —— 状态管理 pinia
    鸿蒙原生应用元服务开发-WebGL网页图形库开发无着色器绘制2D图形
    scau Java综合性实验之Java源程序分析程序
    CSS3 飘动的云和热气球
    C++实战项目-网络编程基础包含TCP详解
    多业务模式下的交易链路探索与实践
    国内代码托管平台Gitee(码云)的入门使用
    BUUCTF--[ACTF2020 新生赛]Include
    rabbitmq 面试题
  • 原文地址:https://blog.csdn.net/qq_41934502/article/details/133322206