文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

C++实现一个简单消息队列的示例详解

2022-12-15 12:02

关注

前言

消息队列在多线程的场景有时会用到,尤其是线程通信跨线程调用的时候,就可以使用消息队列进行通信。C++实现一个能用的消息队列还是比较简单的,只需要一个队列一个互斥变量和一个条件变量,这些在标准库中都有提供。基于曾经写过的项目,总结出来最简单的消息队列的实现将在下文中介绍。

一、如何实现

1、接口定义

一个基本的消息队列只需要3个接口:

(1)推送消息

推送消息即是将消息写入队列,这个通常采用异步实现,推送之后立刻返回。如果要实现Windows的SendMessage则会比较复杂,最好的方式是放到外部实现,消息队列只提供异步推送消息。

void push(const T& msg);    

(2)等待消息

等待队列的消息,这个方法是同步的,只有接收到消息才会返回。

//等待消息
void wait(T& msg);

(3)轮询消息

轮询消息和等待消息一样也是接收消息,只是无论是否接收到消息轮询消息会立刻返回。轮询消息也是有一定的使用场景,尤其是接收消息线程需要一定的调度逻辑时就需要轮询消息避免线程堵塞。

bool poll(T& msg);

2、用到的对象

(1)队列

我们使用一个队列来存放消息

#include<queue>
std::queue<T> _queue;

(2)互斥变量

使用一个互斥变量确保队列的读写线程安全

#include<mutex>
std::mutex _mtx;

(3)条件变量

采用条件变量结合互斥变量实现消息的等待和通知。

#include<condition_variable>
std::condition_variable _cv;

3、基本流程

线程通信

二、完整代码

采用泛型实现,消息类型可以自定义。

#include<mutex>
#include<condition_variable>
#include<queue>
/// <summary>
/// 消息队列
/// </summary>
/// <typeparam name="T">消息类型</typeparam>
template<class T> class MessageQueue {
public:
    /// <summary>
    /// 推入消息
    /// </summary>
    /// <param name="msg">消息对象</param>
    void push(const T& msg) {
        std::unique_lock<std::mutex>lck(_mtx);
        _queue.push(msg);
        _cv.notify_one();
    }
    /// <summary>
    /// 轮询消息
    /// </summary>
    /// <param name="msg">消息对象</param>
    /// <returns>是否接收到消息</returns>
    bool poll(T& msg) {
        std::unique_lock<std::mutex>lck(_mtx);
        if (_queue.size())
        {
            msg = _queue.front();
            _queue.pop();
            return true;
        }
        return false;
    }
    /// <summary>
    /// 等待消息
    /// </summary>
    /// <param name="msg">消息对象</param>
    void wait(T& msg) {
        std::unique_lock<std::mutex>lck(_mtx);
        while (!_queue.size()) _cv.wait(lck);
        msg = _queue.front();
        _queue.pop();
    }
    //队列长度
    size_t size() {
        std::unique_lock<std::mutex>lck(_mtx);
        return _queue.size();
    }
private:
    //队列
    std::queue<T> _queue;
    //互斥变量
    std::mutex _mtx;
    //条件变量
    std::condition_variable _cv;
};

三、使用示例

线程通信

等待消息

#include<thread>
//自定义消息对象
class MyMessage {
public:
    int type;
    void* param1;
    void* param2;
};
int main(int argc, char* argv[])
{
    //初始化消息队列
    MessageQueue<MyMessage> mq;
    //启动线程
    std::thread t1([&]() {
        MyMessage msg;
        while (1) {
            //等待队列的消息
            mq.wait(msg);
            printf("receive message type:%d\n", msg.type);
            if (msg.type == 1001)
                break;
        }
        printf("thread exited\n");
        });
    //发送消息给线程
    MyMessage msg;
    printf("send number message to thread.1001 exit\n");
    while (1)
    {
        scanf("%d", &msg.type);
        mq.push(msg);
        if (msg.type == 1001)
            break;
    }
    t1.join();
    return 0;
}

轮询消息

#include<thread>
//自定义消息对象
class MyMessage {
public:
    int type;
    void* param1;
    void* param2;
};
int main(int argc, char* argv[])
{
    //初始化消息队列
    MessageQueue<MyMessage> mq;
    //启动线程
    std::thread t1([&]() {
        MyMessage msg;
        while (1) {
            //轮询队列的消息
            if (mq.poll(msg))
            {
                printf("receive message type:%d\n", msg.type);
                if (msg.type == 1001)
                    break;
            }
            std::this_thread::sleep_for(std::chrono::milliseconds(10));
        }
        printf("thread exited\n");
        });
    //发送消息给线程
    MyMessage msg;
    printf("send number message to thread.1001 exit\n");
    while (1)
    {
        scanf("%d", &msg.type);
        mq.push(msg);
        if (msg.type == 1001)
            break;
    }
    t1.join();
    return 0;
}

总结

以上就是今天要讲的内容,实现一个简单消息队列还是比较容易的,尤其是c++有标准库支持的情况下,也能满足大部分使用场景,比如实现线程切换或者async、await底层就需要用到消息队列。写这篇博文的主要目的也是用于记录,以后需要用到的时候可直接网上拷贝。

到此这篇关于C++实现一个简单消息队列的示例详解的文章就介绍到这了,更多相关C++消息队列内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯