C++多线程通信

news2025/1/20 5:58:23

多线程通信

  • 引言
  • 区别
    • 活锁
      • 什么是活锁
      • 与死锁的区别
      • 如何避免活锁
  • 多线程通信
    • 示例
    • 运行结果

引言

多线程同步与多线程通信实质上是两种相互关联但又不完全相同的东西。本文注重多线程同步与多线程通信的区别,同时重点讲述多线程通信中的消息队列。

区别

多线程通信通常指的是线程之间交换信息或数据的方式。线程之间可能需要共享数据、发送消息或信号,以便协调它们的行为或实现某种协作。

多线程同步则是关于控制或协调多个线程的执行顺序,以确保它们能够正确地共享资源或执行特定的任务序列。同步机制用于防止数据竞争(data race)和其他并发问题,例如死锁(deadlock)和活锁(livelock)。

活锁

关于活锁记录一下。

什么是活锁

活锁(Livelock)是一种特殊的并发现象,它发生在多个进程或线程在尝试访问共享资源时,由于某种条件没有满足,导致它们不断重复尝试访问,但始终无法成功。这些进程或线程在活锁状态下会不断地改变状态,但它们并没有真正地“前进”,因此得名“活锁”。

活锁可以认为是一种特殊的饥饿现象,因为进程或线程虽然没有被阻塞,但它们始终无法访问所需的资源,从而导致它们无法继续执行。

与死锁的区别

与死锁(Deadlock)不同的是,活锁中的实体(进程或线程)并没有停止运行,而是不断尝试访问资源,但由于某些条件没有满足(例如,资源被其他进程占用,或者进程间的通信出现问题),它们始终无法成功访问。因此,活锁有可能自行解开,只要满足了相关条件,进程或线程就可以成功访问资源。

如何避免活锁

一种简单方法是采用先来先服务的策略,当多个事务请求封锁同一数据对象时,封锁子系统按请求封锁的先后次序对事务排队,数据对象上的锁一旦释放就批准申请队列中第一个事务获得锁。这样可以确保每个进程或线程都有机会访问资源,从而避免活锁的发生。

多线程通信

多线程通信的方式有:
共享内存、消息队列、互斥锁、条件变量、信号量、future和promise。
互斥锁、条件变量、信号量、future和promise的使用可以查阅:
多线程同步(上)
多线程同步(下)
本文主要讲解消息队列。

示例

下面是一个简单的实现示例。实现的是两个线程同时对一个消息队列操作,消息队列最大容纳5个数据,一个生产者线程向里存数据,一个消费者线程向外取数据,总共循环十次。

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

using namespace std;
constexpr int Max = 5;
constexpr int g_count = 10;

// 消息队列类  
class MessageQueue {
public:
    // 入队操作  
    void Enqueue(int message) {
        std::unique_lock<std::mutex> lock(mutex_);
        cond_.wait(lock, [&] {return queue_.size() < 5; });
        queue_.push(message);
        cond_.notify_one(); // 通知等待的线程  
    }

    // 出队操作  
    bool Dequeue(int& message) {
        std::unique_lock<std::mutex> lock(mutex_);
        cond_.wait(lock, [this] { return !queue_.empty(); }); // 等待直到队列不为空  
        message = queue_.front();
        queue_.pop();
        cond_.notify_all();
        return true;
    }

private:
    std::queue<int> queue_;
    std::mutex mutex_;
    std::condition_variable cond_;
    bool m_exit = false;
};

// 生产者线程函数  
void Producer(MessageQueue& queue) {
    for (int i = 0; i < g_count; ++i) {
        std::cout << "Producing message: " << i << std::endl;
        queue.Enqueue(i);
        std::this_thread::sleep_for(std::chrono::milliseconds(100)); // 模拟耗时操作  
    }
}

// 消费者线程函数  
void Consumer(MessageQueue& queue) {
    int message;
    for (size_t i = 0; i < g_count; i++)
    {
        if (queue.Dequeue(message)) {
            std::cout << "Consuming message: " << message << std::endl;
            std::this_thread::sleep_for(std::chrono::milliseconds(100)); // 模拟耗时操作  
        }
    } 
}

int main() {
    MessageQueue queue;

    // 创建生产者线程  
    std::thread producerThread(Producer, std::ref(queue));

    // 创建消费者线程  
    std::thread consumerThread(Consumer, std::ref(queue));

    // 等待生产者线程完成  
    producerThread.join();

    // 等待消费者线程完成  
    consumerThread.join();

    return 0;
}

运行结果

在这里插入图片描述
上述的示例只是一个非常简单的示例。重在理解消息队列在两个线程之间的传递数据。
上述这个例子中也有共享内存的存在,其中队列里的数据可供一个线程写入,也供另一个线程读取。
本示例比较简单,后面再更新吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1475376.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Springboot+vue的考务报名平台(有报告)。Javaee项目,springboot vue前后端分离项目。

演示视频&#xff1a; Springbootvue的考务报名平台&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot vue前后端分离项目。 项目介绍&#xff1a; 本文设计了一个基于Springbootvue的前后端分离的考务报名平台&#xff0c;采用M&#xff08;model&#xff0…

SpringMVC了解

1.springMVC概述 Spring MVC&#xff08;Model-View-Controller&#xff09;是基于 Java 的 Web 应用程序框架&#xff0c;用于开发 Web 应用程序。它通过将应用程序分为模型&#xff08;Model&#xff09;、视图&#xff08;View&#xff09;和控制器&#xff08;Controller&a…

Flink SQL 中的流式概念:状态算子

博主历时三年精心创作的《大数据平台架构与原型实现&#xff1a;数据中台建设实战》一书现已由知名IT图书品牌电子工业出版社博文视点出版发行&#xff0c;点击《重磅推荐&#xff1a;建大数据平台太难了&#xff01;给我发个工程原型吧&#xff01;》了解图书详情&#xff0c;…

【MySQL面试复习】了解过索引吗?(索引的底层原理)/B 树和B+树的区别是什么?

系列文章目录 在MySQL中&#xff0c;如何定位慢查询&#xff1f; 发现了某个SQL语句执行很慢&#xff0c;如何进行分析&#xff1f; 系列文章目录了解过索引吗&#xff1f;(索引的底层原理)B 树和B树的区别是什么&#xff1f; 了解过索引吗&#xff1f;(索引的底层原理) 如果没…

K8S之Deployment的介绍和使用

Deployment的理论和实操 Deployment控制器&#xff1a;概念、原理解读概述工作原理 编写Deployment资源清单文件使用案例&#xff1a;创建一个web站点Deployment管理pod&#xff1a;扩容、缩容通过deployment管理应用&#xff0c;实现扩容&#xff0c;把副本数变成3通过deploym…

面试必问但日常不愿意看的题

1&#xff0c;做道 this 相关的题&#xff0c;看你对 js 的 this 掌握的如何2&#xff0c;BFC 这样答才完美 1&#xff0c;什么是 BFC&#xff1f;其规则是什么&#xff1f;2&#xff0c;如何触发 BFC3&#xff0c;BFC 到底可以解决什么问题呢3&#xff0c;作用域4&#xff0c;…

疫情物资智能管理:Java与SpringBoot的实践

✍✍计算机毕业编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java、…

Python算法100例-2.6 分糖果

完整源代码项目地址&#xff0c;关注博主私信源代码后可获取 1.问题描述2.问题分析3.算法设计4.确定程序框架5.完整的程序6.运行结果 1&#xff0e;问题描述 10个小孩围成一圈分糖果&#xff0c;老师分给第1个小孩10块&#xff0c;第2个小孩2块&#xff0c;第3个小孩8块&…

区块链智能合约开发

一.区块链的回顾 1.区块链 区块链实质上是一个去中心化、分布式的可进行交易的数据库或账本 特征: 去中心化&#xff1a;简单来说&#xff0c;在网络上一个或多个服务器瘫痪的情况下&#xff0c;应用或服务仍然能够持续地运行&#xff0c;这就是去中心化。服务和应用部署在…

SpringMVC 学习(八)之文件上传与下载

目录 1 文件上传 2 文件下载 1 文件上传 SpringMVC 对文件的上传做了很好的封装&#xff0c;提供了两种解析器。 CommonsMultipartResolver&#xff1a;兼容性较好&#xff0c;可以兼容 Servlet3.0 之前的版本&#xff0c;但是它依赖了 commons-fileupload …

Eavesdropping(窃听机制)在机器学习中的用法

1. 简单翻译 考虑一个对任务 T 和 T’ 有用的特征 F&#xff0c;它在学习 T 时很容易学习&#xff0c;但在学习 T’ 时很难学习&#xff0c;因为 T’ 以更复杂的方式使用 F。网络学习 T 将学习 F&#xff0c;但网络学习 T’ 可能不会。如果网络学习 T’ 也学习 T&#xff0c;T…

每日汇评:黄金多头拒绝在美国宏观数据发布前放弃

周三早些时候&#xff0c;金价买家再次测试两周高点 2041美元&#xff1b; 美元延续反弹&#xff0c;但疲弱的国债收益率可能限制其上涨空间&#xff1b; 由于金价等待美国数据&#xff0c;4小时图表技术面似乎具有建设性&#xff1b; 金价正在复制周二亚洲交易中的价格走势&am…

江科大stm32 定时器 TIM输出比较--学习笔记

这几天遇到输出比较相关的问题&#xff0c;于是来学习下TIM输出比较部分知识点&#xff01; 输出比较简介 CNT是计数器的值&#xff0c;CCR寄存器是捕获/ 比较寄存器 简单的讲&#xff0c;输出比较就是用来输出PWM波形。 PWM简介 占空比&#xff1a;高电平占一个周期的比例。…

VScode打开keil5软件的内容

VScode想要打开keil5软件的内容&#xff0c;需要在此引入 具体可参考&#xff1a; VS Code环境下编辑、编译、下载Keil工程代码

本届挑战赛亚军方案:面向微服务架构系统中无标注、多模态运维数据的异常检测、根因定位与可解释性分析

CheerX团队来自于南瑞研究院系统平台研发中心&#xff0c;中心主要从事NUSP电力自动化通用软件平台的关键技术研究与软件研发。 选题分析 图1 研究现状 本次CheerX团队的选题紧密贴合了目前的运维现状。实际运维中存在多种问题导致运维系统的不可用。比如故障发生时&#xff…

【Maven】Maven 基础教程(一):基础介绍、开发环境配置

Maven 基础教程&#xff08;一&#xff09;&#xff1a;基础介绍、开发环境配置 1.Maven 是什么1.1 构建1.2 依赖 2.Maven 开发环境配置2.1 下载安装2.2 指定本地仓库2.3 配置阿里云提供的镜像仓库2.4 配置基础 JDK 版本2.5 配置环境变量 1.Maven 是什么 Maven 是 Apache 软件…

Docker部署Portainer图形化管理工具

文章目录 前言1. 部署Portainer2. 本地访问Portainer3. Linux 安装cpolar4. 配置Portainer 公网访问地址5. 公网远程访问Portainer6. 固定Portainer公网地址 前言 Portainer 是一个轻量级的容器管理工具&#xff0c;可以通过 Web 界面对 Docker 容器进行管理和监控。它提供了可…

2024最后一次Java面试,java高级开发面试经验

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

Python Skip-Gram代码实战,Skip-Gram代码超简单讲解和步骤拆解,Word2vec代码构建思路,Skip-Gram代码实例,模板套用

1. Skip-Gram介绍 Skip-gram模型是Word2Vec模型的一种训练方法&#xff0c;它的目标是通过目标词预测上下文词。Skip-gram模型通过神经网络结构来学习每个单词的向量表示。 在Skip-gram模型中&#xff0c;每个单词被表示为一个固定维度的向量&#xff0c;该向量称为嵌入向量或词…

AStar算法(大物件寻路)

前言 A星(物件大小为一格)寻路&#xff0c;都很熟悉了吧&#xff0c;网上源码一堆&#xff0c;随便抄; 这章需要讲述 大物件的A星寻路&#xff0c;何为大物件&#xff0c;就是 比如 物件 为4个格子&#xff1b; 这样&#xff0c;原来的A星 没法直接用了&#xff0c;必须得改装…