新闻中心

C++如何实现一个阻塞队列_C++并发队列设计与阻塞机制实现

2025-11-23
浏览次数:
返回列表
阻塞队列是C++生产者-消费者模型的核心,通过互斥锁与条件变量实现线程安全和阻塞操作,支持有界/无界队列设计,配合wait、notify机制协调生产与消费,避免竞争与资源浪费。

c++如何实现一个阻塞队列_c++并发队列设计与阻塞机制实现

在C++多线程编程中,阻塞队列是实现生产者-消费者模型的关键组件。它允许多个线程安全地共享数据,当队列为空时,消费者线程自动等待;当队列满时,生产者线程等待,直到有空间可用。这种机制能有效避免资源浪费和竞争条件。

阻塞队列的基本设计思路

一个阻塞队列需要满足以下几点:

  • 线程安全:多个线程可以同时访问队列而不导致数据竞争
  • 阻塞操作:取元素时若队列为空,则调用线程应被阻塞;插入时若队列满,则阻塞直到有空间
  • 可配置容量:支持有界队列(固定大小)或无界队列

核心依赖是互斥锁(std::mutex)条件变量(std::condition_variable),用于同步线程和触发唤醒。

使用标准库实现阻塞队列

下面是一个基于std::queuestd::mutexstd::condition_variable的简单阻塞队列实现:

#include <queue>
#include <mutex>
#include <condition_variable>
#include <thread>

template<typename T>
class BlockingQueue {
private:
    std::queue<T> queue_;
    mutable std::mutex mtx_;
    std::condition_variable not_empty_;
    std::condition_variable not_full_;
    size_t capacity_;

public:
    explicit BlockingQueue(size_t cap = 1000) : capacity_(cap) {}

    void put(T item) {
        std::unique_lock<std::mutex> lock(mtx_);
        not_full_.wait(lock, [this] { return queue_.size() < capacity_; });
        queue_.push(std::move(item));
        not_empty_.notify_one();
    }

    T take() {
        std::unique_lock<std::mutex> lock(mtx_);
        not_empty_.wait(lock, [this] { return !queue_.empty(); });
        T value = std::move(queue_.front());
        queue_.pop();
        not_full_.notify_one();
        return value;
    }

    bool empty() const {
        std::lock_guard<std::mutex> lock(mtx_);
        return queue_.empty();
    }

    size_t size() const {
        std::lock_guard<std::mutex> lock(mtx_);
        return queue_.size();
    }
};

说明:

PictoGraphic PictoGraphic

AI驱动的矢量插图库和插图生成平台

PictoGraphic 133 查看详情 PictoGraphic
  • put() 在插入前等待队列不满,插入后通知等待的消费者
  • take() 等待队列非空,取出后通知生产者
  • 使用mutable修饰互斥锁,以便在const成员函数中加锁
  • 采用std::unique_lock配合条件变量,支持等待时释放锁

线程安全与性能优化建议

虽然上述实现是线程安全的,但在高并发场景下仍有改进空间:

  • 使用std::deque替代std::queue以支持高效首尾操作
  • 添加超时版本的put/take,如bool try_put(T, timeout),避免无限等待
  • 对无界队列,只保留not_empty_条件变量即可
  • 考虑使用无锁队列(如基于CAS操作)提升性能,但复杂度显著增加

实际使用示例

以下是一个简单的生产者-消费者测试:

int main() {
    BlockingQueue<int> bq(5);

    std::thread producer([&bq]() {
        for (int i = 0; i < 10; ++i) {
            bq.put(i);
            std::cout << "Produced: " << i << "\n";
        }
    });

    std::thread consumer([&bq]() {
        for (int i = 0; i < 10; ++i) {
            int val = bq.take();
            std::cout << "Consumed: " << val << "\n";
        }
    });

    producer.join();
    consumer.join();
    return 0;
}

这个例子展示了两个线程通过阻塞队列交换数据,无需额外同步逻辑。

基本上就这些。掌握阻塞队列的实现,有助于深入理解C++并发编程中的同步机制。不复杂但容易忽略细节,比如条件变量的谓词判断和notify的调用时机。

以上就是C++如何实现一个阻塞队列_C++并发队列设计与阻塞机制实现的详细内容,更多请关注其它相关文章!


# 为空  # seo多项选  # 江西济南网站建设  # 葫芦岛网站推广方法  # 四川网站建设方案优化公司  # 衡水网络推广网站优化  # seo年终述的明年规划  # qq互点++seo有问题吗  # 网站优化排行榜在线咨询  # 宝鸡钛金属网站优化  # seo浏阳招聘信息  # 边缘  # 资源浪费  # ai  # 互斥  # 无界  # 多个  # 如何实现  # 多线程  # 是一个  # 游戏开发  # 标准库  # 同步机制  # 无锁  # 并发编程  # c++ 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: J*a递归快速排序中静态变量导致数据累积问题的解决方案  树莓派传感器触发:通过Twilio API发送WhatsApp消息教程  如何优雅地扩展SprykerGlue后端API授权逻辑,使用spryker/glue-backend-api-application-authorization-connector-extension  百度浏览器字体显示异常偏小_百度浏览器字体渲染修复方案  QQ邮箱官方登录入口_QQ邮箱网页版快捷使用平台  PDF文件体积过大处理_PDF压缩技巧详解  QQ邮箱官方网站登录入口_QQ邮箱网页版在线使用  深入理解J*aScript中的B样条曲线与节点向量生成  J*aScript Promise链中如何正确终止后续.then执行并处理错误  谷歌浏览器无痕模式怎么开 Chrome开启无痕浏览设置方法【教程】  C++如何实现异步操作_C++11使用std::future和std::async进行异步编程  在J*a中如何隐藏复杂性_使用门面模式组织对象交互  圆通快递查询实时追踪 圆通物流包裹状态快速查看  小红书网页版入口链接分享 小红书官网直接进  顺丰快递查询系统 官方正版查询入口  2025俄罗斯Yandex最新入口 官方网站地址及浏览器下载指南  mysql通配符支持数字匹配吗_mysql通配符能否用于数字匹配的解析  PostgreSQL海量数据高效导入策略:Python与Django实践指南  mcjs网页版在线存档 mcjs云存档登录入口  快速CSGO开箱网站指南 CSGO开箱平台推荐  sublime如何配置Go语言开发环境_sublime搭建Golang编译运行系统  Windows电脑怎么截图最方便_系统自带截图工具的5种神仙用法【技巧】  创客贴用户入口官网登录 创客贴网页版电脑版系统  天眼查怎么看公司融资情况 天眼查企业融资历史查询步骤【攻略】  如何将HTML表格多行数据保存到Google Sheets  微博网页版首页入口 微博电脑端官网登录链接  J*aScript map 迭代中检测空数组元素的有效方法  Fabric Mod开发:在1.19.3+版本中正确添加自定义物品并管理物品组  age动漫网站入口 age动漫官网直接访问入口  邮政编码查询不到怎么办_邮政编码查询不到的常见原因与对策  三星GalaxyZFold5怎样在相册制作折叠屏分镜_iPhone三星GalaxyZFold5相册制作折叠屏分镜【创意编辑】  Gmail邮箱申请注册直达_Gmail邮箱免费注册PC版官网入口2025  照顾宝贝2小游戏免费秒玩入口  使用Pandas转换并合并DataFrame:多列映射至统一结构  如何设置Windows Defender的定时扫描_计划任务实现自动杀毒【安全】  没有大陆身份证/银行卡如何实名微信? 亲测有效的几种方法分享  大象笔记网页版入口 印象笔记网页版登录入口  Discord Slash 命令响应超时问题的异步解决方案  漫蛙MANWA漫画主页官方入口 漫蛙漫画最新在线阅读地址  抖音商城签到领现金是真的吗_抖音商城签到奖励与提现说明  快手极速版在线观看 官方网页版登录地址  邮政快递包裹最新位置 邮政快递实时追踪入口  12306选座怎么选到特殊座位_12306特殊座位选择注意事项  j*a toString()的覆盖  葱吃多了会怎样 葱吃多了会伤胃吗  怎么在html里运行vbs脚本_html中运行vbs脚本方法【教程】  Safari怎么安装扩展程序 浏览器插件安装与管理方法【详解】  微博网页版怎么开启两步验证_微博网页版账号安全两步验证设置方法  怎样在Excel中做仪表盘_Excel仪表盘设计与关键指标展示方法  PHP中获取MongoDB服务器运行时间(Uptime)的专业指南 

搜索