C++ 实现一个消息队列


前言

消息队列在多线程的场景有时会用到,尤其是线程通信跨线程调用的时候,就可以使用消息队列进行通信。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、基本流程

(1)、线程通信

在这里插入图片描述


二、完整代码

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

#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;
};

三、使用示例

1、线程通信

(1)、等待消息

#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;
}

(2)、轮询消息

#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底层就需要用到消息队列。写这篇博文的主要目的也是用于记录,以后需要用到的时候可直接网上拷贝。

猜你喜欢

转载自blog.csdn.net/u013113678/article/details/127585569
今日推荐