广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >怎么在C++项目中利用priority_queue自定义排序
  • 684
分享到

怎么在C++项目中利用priority_queue自定义排序

2023-06-06 17:06:11 684人浏览 安东尼
摘要

这篇文章给大家介绍怎么在c++项目中利用priority_queue自定义排序,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。首先,无论 priority_queue 中存储的是基础数据类型(int、double 等),

这篇文章给大家介绍怎么在c++项目中利用priority_queue自定义排序,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。

首先,无论 priority_queue 中存储的是基础数据类型(int、double 等),还是 string 类对象或者自定义的类对象,都可以使用函数对象的方式自定义排序规则。例如:

#include<iOStream>#include<queue>using namespace std;//函数对象类template <typename T>class cmp{public:  //重载 () 运算符  bool operator()(T a, T b)  {    return a > b;  }};int main(){  int a[] = { 4,2,3,5,6 };  priority_queue<int,vector<int>,cmp<int> > pq(a,a+5);  while (!pq.empty())  {    cout << pq.top() << " ";    pq.pop();  }  return 0;}

运行结果为:
2 3 4 5 6

注意,C++ 中的 struct 和 class 非常类似,前者也可以包含成员变量和成员函数,因此上面程序中,函数对象类 cmp 也可以使用 struct 关键字创建:

struct cmp{  //重载 () 运算符  bool operator()(T a, T b)  {    return a > b;  }};

可以看到,通过在 cmp 类(结构体)重载的 () 运算符中自定义排序规则,并将其实例化后作为 priority_queue 模板的第 3 个参数传入,即可实现为 priority_queue 容器适配器自定义比较函数。

除此之外,当 priority_queue 容器适配器中存储的数据类型为结构体或者类对象(包括 string 类对象)时,还可以通过重载其 > 或者 < 运算符,间接实现自定义排序规则的目的。

注意,此方式仅适用于 priority_queue 容器中存储的为类对象或者结构体变量,也就是说,当存储类型为类的指针对象或者结构体指针变量时,此方式将不再适用,而只能使用函数对象的方式。

要想彻底理解这种方式的实现原理,首先要搞清楚 std::less<T> 和 std::greater<T> 各自的底层实现。实际上,<function> 头文件中的 std::less<T> 和 std::greater<T> ,各自底层实现采用的都是函数对象的方式。比如,std::less<T> 的底层实现代码为:

template <typename T>struct less {  //定义新的排序规则  bool operator()(const T &_lhs, const T &_rhs) const {    return _lhs < _rhs;  }};

std::greater<T> 的底层实现代码为:

template <typename T>struct greater {  bool operator()(const T &_lhs, const T &_rhs) const {    return _lhs > _rhs;  }};

可以看到,std::less<T> 和 std::greater<T> 底层实现的唯一不同在于,前者使用 < 号实现从大到小排序,后者使用 > 号实现从小到大排序。

那么,是否可以通过重载 < 或者 > 运算符修改 std::less<T> 和 std::greater<T> 的排序规则,从而间接实现自定义排序呢?答案是肯定的,举个例子:

#include<queue>#include<iostream>using namespace std;class node {public:  node(int x = 0, int y = 0) :x(x), y(y) {}  int x, y;};//新的排序规则为:先按照 x 值排序,如果 x 相等,则按 y 的值排序bool operator < (const node &a, const node &b) {  if (a.x > b.x) return 1;  else if (a.x == b.x)    if (a.y >= b.y) return 1;  return 0;}int main() {  //创建一个 priority_queue 容器适配器,其使用默认的 vector 基础容器以及 less 排序规则。  priority_queue<node> pq;  pq.push(node(1, 2));  pq.push(node(2, 2));  pq.push(node(3, 4));  pq.push(node(3, 3));  pq.push(node(2, 3));  cout << "x y" << endl;  while (!pq.empty()) {    cout << pq.top().x << " " << pq.top().y << endl;    pq.pop();  }  return 0;}

输出结果为:
x y
1 2
2 2
2 3
3 3
3 4

可以看到,通过重载 < 运算符,使得 std::less<T> 变得适用了。
读者还可以自行尝试,通过重载 > 运算符,赋予 std::greater<T> 和之前不同的排序方式。

当然,也可以以友元函数或者成员函数的方式重载 > 或者 < 运算符。需要注意的是,以成员函数的方式重载 > 或者 < 运算符时,该成员函数必须声明为 const 类型,且参数也必须为 const 类型,至于参数的传值方式是采用按引用传递还是按值传递,都可以(建议采用按引用传递,效率更高)。

例如,将上面程序改为以成员函数的方式重载 < 运算符:

class node {public:  node(int x = 0, int y = 0) :x(x), y(y) {}  int x, y;  bool operator < (const node &b) const{    if ((*this).x > b.x) return 1;    else if ((*this).x == b.x)      if ((*this).y >= b.y) return 1;    return 0;  }};

同样,在以友元函数的方式重载 < 或者 > 运算符时,要求参数必须使用 const 修饰。例如,将上面程序改为以友元函数的方式重载 < 运算符。例如:

class node {public:  node(int x = 0, int y = 0) :x(x), y(y) {}  int x, y;  friend bool operator < (const node &a, const node &b);};//新的排序规则为:先按照 x 值排序,如果 x 相等,则按 y 的值排序bool operator < (const node &a, const node &b){  if (a.x > b.x) return 1;  else if (a.x == b.x)    if (a.y >= b.y) return 1;  return 0;}

关于怎么在C++项目中利用priority_queue自定义排序就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

--结束END--

本文标题: 怎么在C++项目中利用priority_queue自定义排序

本文链接: https://www.lsjlt.com/news/248168.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么在C++项目中利用priority_queue自定义排序
    这篇文章给大家介绍怎么在C++项目中利用priority_queue自定义排序,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。首先,无论 priority_queue 中存储的是基础数据类型(int、double 等),...
    99+
    2023-06-06
  • 怎么在c#项目中自定义MarkupExtension
    这篇文章给大家介绍怎么在c#项目中自定义MarkupExtension,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Markup Extension,顾名思义,就是对xaml的扩展,在XAML中,规定如果属性以{}开始...
    99+
    2023-06-06
  • JAVA在项目中怎么自定义异常
    这篇文章主要讲解了“JAVA在项目中怎么自定义异常”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JAVA在项目中怎么自定义异常”吧!JAVA项目中自定义异常1.数据返回处理类@Datapub...
    99+
    2023-06-20
  • 如何在C#项目中使用WPF自定义按钮
    今天就跟大家聊聊有关如何在C#项目中使用WPF自定义按钮,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。实现效果使用图片做按钮背景;自定义鼠标进入时效果;自定义按压效果;自定义禁用效果...
    99+
    2023-06-06
  • php中怎么利用uasort()自定义排序 多维数组
    php中怎么利用uasort()自定义排序 多维数组,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。php内置的排序函数很多;正反各种排;常用的...
    99+
    2022-10-19
  • PHP中怎么利用usort()函数实现自定义排序
    PHP中怎么利用usort()函数实现自定义排序,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。PHP函数usort()的一个例子,在这个例子中根据它们的长度对数组元素进行排序,...
    99+
    2023-06-17
  • 怎么在java项目中对注解自定义
    这期内容当中小编将会给大家带来有关怎么在java项目中对注解自定义,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。java  自定义注解的实例详解Java的Annotation是在5.0版本之后引...
    99+
    2023-05-31
    java 中对 ava
  • 怎么在java项目中自定义线程池
    怎么在java项目中自定义线程池?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。使用线程池时,可以使用 newCachedThreadPool()或者 newFi...
    99+
    2023-05-31
    java 线程池 ava
  • 在Android项目中怎么自定义阴影效果
    在Android项目中怎么自定义阴影效果?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。Android 自定义阴影效果详解及实例Android5.X中,Google为其增加了...
    99+
    2023-05-31
    android roi 目中
  • 怎么在VS2019中自定义一个项目模板
    怎么在VS2019中自定义一个项目模板?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一、自定义项目模板创建 本次我们来创建一个实现了基础功能的webapi模板 创建自定义的项目...
    99+
    2023-06-14
  • 在iview+vue项目中怎么使用自定义icon图标
    这篇文章主要讲解了“在iview+vue项目中怎么使用自定义icon图标”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“在iview+vue项目中怎么使用自定义icon图标”吧!1. UI设计...
    99+
    2023-06-29
  • 怎么在golang中对Map进行键值自定义排序
    本篇文章为大家展示了怎么在golang中对Map进行键值自定义排序,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。golang适合做什么golang可以做服务器端开发,但golang很适合做日志处理、...
    99+
    2023-06-14
  • 怎么在Java项目中利用QuickSort实现一个快速排序功能
    今天就跟大家聊聊有关怎么在Java项目中利用QuickSort实现一个快速排序功能,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。快速排序-----------------------...
    99+
    2023-05-31
    quicksort java
  • 怎么在Java中对TreeSet进行自定义类型的排序
    怎么在Java中对TreeSet进行自定义类型的排序?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Java可以用来干什么Java主要应用于:1. web开发;2. Androi...
    99+
    2023-06-06
  • 如何在C++中使用STL关联式容器自定义排序规则
    如何在C++中使用STL关联式容器自定义排序规则?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。1) 使用函数对象自定义排序规则#include <iostrea...
    99+
    2023-06-06
  • 怎么在java 项目中使用归并排序算法
    怎么在java 项目中使用归并排序算法?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。归并排序     &nbs...
    99+
    2023-05-31
    java 归并排序 ava
  • 怎么在python中利用namedtuple自定义数据类型
    怎么在python中利用namedtuple自定义数据类型?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。python可以做什么Python是一种编程语言,内置了许多有效的工...
    99+
    2023-06-14
  • 怎么在Android中利用marker自定义一个弹框窗口
    怎么在Android中利用marker自定义一个弹框窗口?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Android是什么Android是一种基于Linux内核的自由及开放源代...
    99+
    2023-06-14
  • 怎么在Java中利用AQS实现一个自定义同步器
    这篇文章主要为大家详细介绍了怎么在Java中利用AQS实现一个自定义同步器,文中示例代码介绍的非常详细,具有一定的参考价值,发现的小伙伴们可以参考一下:Java的特点有哪些Java的特点有哪些1.Java语言作为静态面向对象编程语言的代表,...
    99+
    2023-06-06
  • 怎么在Android中利用构造函数对View进行自定义
    本篇文章给大家分享的是有关怎么在Android中利用构造函数对View进行自定义,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。自定义View有三个构造方法,它们的作用是不同的。...
    99+
    2023-05-31
    android view roi
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作