0

0

C++如何实现带优先级的任务队列?(priority_queue封装)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-20 12:25:02

|

753人浏览过

|

来源于php中文网

原创

std::priority_queue 默认是大顶堆,top() 返回最大值;因其默认使用std::less比较器,若需小值优先(如定时任务),须显式指定std::greater作为第三个模板参数。

c++如何实现带优先级的任务队列?(priority_queue封装)

为什么 std::priority_queue 默认不是“高优先级先出”?

很多人一用就发现:插入 1105top() 却返回 10——以为是“最大堆”,结果业务要的是“数值越小,优先级越高”(比如定时任务按毫秒时间戳排序),这就反了。

根本原因是:std::priority_queue 默认用 std::less<t></t>,底层是大顶堆,top() 返回最大值。它不关心你叫它“优先级队列”,只认比较逻辑。

  • 要实现“小值优先”,得显式传入 std::greater<int></int> 作为第三个模板参数
  • 自定义类型必须提供 operator 或传入仿函数,否则编译报错:<code>invalid operands to binary expression
  • 别在运行时“反转逻辑”,比如手动存 -timestamp——容易溢出、可读性差、调试时 timestamp 显示为负数

如何让自定义任务结构体正确参与优先级比较?

直接把 struct Task 塞进 std::priority_queue 会失败,因为默认没有定义大小关系。常见错误是只重载了 operator== 或写了个 compare() 成员函数但没被调用。

最稳妥的做法是提供 operator,且语义清晰:返回 <code>true 表示左边“优先级更低”(即应该排在右边之后)。

立即学习C++免费学习笔记(深入)”;

Musico
Musico

Musico 是一个AI驱动的软件引擎,可以生成音乐。 它可以对手势、动作、代码或其他声音做出反应。

下载
struct Task {
    int priority;  // 数值越小,越该先执行
    std::string name;
    bool operator<(const Task& rhs) const {
        return priority > rhs.priority;  // 注意:这里是 >,不是 <
    }
};
  • 上面的 operator 实际构建的是小顶堆:当 <code>this->priority > rhs.priority,说明 this 该排后面,所以 this 为 <code>false,符合 std::less 的预期
  • 如果用仿函数替代,记得第三个模板参数写成 std::priority_queue<task std::vector>, Compare></task>,别漏掉容器类型
  • 避免在 operator 里做耗时操作(如字符串比较、IO),否则 <code>push()pop() 性能直线下滑

std::priority_queue 能否动态修改已入队任务的优先级?

不能。这是它和真正“可更新堆”(如 Boost.Heap 的 fibonacci_heap)最本质的区别:std::priority_queue 是只读接口封装,底层 std::vector 不暴露索引,也没有 update()erase() 方法。

一旦任务入队,它的位置就被固定;想改优先级,唯一办法是标记旧任务为无效,再 push 一个新任务。

  • 配合使用 std::shared_ptr<task></task> + 状态标志位(如 bool valid = true),pop() 后检查再处理
  • 别试图用 std::make_heap 手动维护底层容器——std::priority_queue 不保证公开其内部 vector,不同 STL 实现可能加 padding 或私有成员,强行访问会未定义行为
  • 如果业务频繁修改优先级(如网络请求重试倒计时动态调整),建议换用 std::set 或第三方库,别硬撑

多线程环境下直接用 std::priority_queue 安全吗?

不安全。它没有任何线程同步机制。两个线程同时 push() 可能导致底层 vector 重分配时数据错乱,top() + pop() 组合也不是原子操作。

最轻量的修复方式是套一层互斥锁,但要注意粒度:

  • 不要只锁 push()pop(),也要锁 empty()size()——它们同样读底层容器
  • 避免在锁内做耗时操作(比如处理任务本身),否则整个队列阻塞,吞吐暴跌
  • 如果需要等待非空再取任务,别用 while+sleep 轮询,改用 std::condition_variable 配合锁

复杂点在于:优先级队列的“唤醒”逻辑比普通队列更敏感——新入队的高优任务,应该立刻打断当前等待线程,而不是等下一次通知。这点很多简单封装会忽略。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Sass和less的区别
Sass和less的区别

Sass和less的区别有语法差异、变量和混合器的定义方式、导入方式、运算符的支持、扩展性等。本专题为大家提供Sass和less相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.10.12

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

790

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

576

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

254

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

420

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

594

2023.08.10

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

776

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

247

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 9.8万人学习

C 教程
C 教程

共75课时 | 4.8万人学习

C++教程
C++教程

共115课时 | 18.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号