关于linux中消息队列的使用

以前对于linux消息队列有过大概的了解,在《uinx高级系统编程》这本书里面提到过。但是具体使用不是很了解,最近再熟悉appserver和tws的框架,这里面使用了linux的消息队列,于是乎进行了系统的了解。消息队列如下

int msgget(key_t key, int msgflg);
int msgctl(int msqid, int cmd, struct msqid_ds *buf);
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);

说明:

msgget()函数用来创建消息队列,key大于0的32位整数:视参数msgflg来确定操作。通常要求此值来源于ftok返回的IPC键值,我自己操作了一下,只要不与别的消息队列重复就行了,成功后返回消息队列msgId

msgsnd()和msgrcv()是发送消息和接受消息的函数,这里要特别说明一下,参数msgp,必须要使用如下结构

typedef struct
{
   long msg_type;
   char msg_txt[size]; // 后面的结构可以自己定义
}msg;
这个msg_type一定要赋值你想要的消息类型(当初以为msgrcv()msgtyp已经指定了参数,不要赋值了,结果收到消息乱序了)。其次msgsz这个参数也要注意。是你接受和发送数据的缓冲区大小(不包括sizeof(msg_type)),如果在msgrcv()中msgsz小于消息的长度,则返回-1(失败)。我猜想这就是为撒需要msg_type的原因,消息块结构(具体有时间去内核源码这个实现部分)。送上测试测试代码

#include "msg_quque.h"

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <errno.h>
#include <string.h>

#include <stdio.h>

typedef struct tagMSG_HEADER
{
	long msg_type;
	int body_len;
	char body[0];
}MSG_HEADER;

CMsgQue *CMsgQue::CreateMsgQueObj(int msg_key)
{
	// 创建消息队列
	int msg_id;
	CMsgQue *pmsg_que = NULL;

	msg_id = msgget(msg_key, IPC_CREAT | 0777);
	if (msg_id == -1)
	{
		printf("msgget() Failed!errno=%d,errstring=%s", errno, strerror(errno));
		return pmsg_que;
	}

	pmsg_que = new CMsgQue(msg_id);

	return pmsg_que;
}

int CMsgQue::Get(char *pmsg_buf, int buf_len, int msg_type)
{
	int size = sizeof(MSG_HEADER) + buf_len;
	MSG_HEADER *pmsg_header = (MSG_HEADER *)new char[size];

	//pmsg_header->msg_type = msg_type;// 这里一定要特别注意
	pmsg_header->body_len = buf_len;

	int ret = msgrcv(m_msg_id, pmsg_header, buf_len, msg_type, 0);
	if (ret == -1)
	{
		printf("msgrcv() Failed!errno=%d,errstring=%s\n", errno, strerror(errno));
		return ret;
	}

	memcpy(pmsg_buf, pmsg_header->body, buf_len);
	delete pmsg_header;

	return ret;
}

int CMsgQue::Put(const char *pmsg_buf, int buf_len, int msg_type)
{
	int size = sizeof(MSG_HEADER) + buf_len;
	MSG_HEADER *pmsg_header = (MSG_HEADER *)new char[size];

	pmsg_header->msg_type = msg_type;
	pmsg_header->body_len = buf_len;
	memcpy(pmsg_header->body, pmsg_buf, buf_len);

	int ret = msgsnd(m_msg_id, pmsg_header, size - sizeof(long), 0);
	if (ret == -1)
	{
		printf("msgsend() Failed!errno=%d,errstring=%s\n", errno, strerror(errno));
		return ret;
	}
	delete pmsg_header;

	return ret;
}

void CMsgQue::Release()
{
	msgctl( msqid, IPC_RMID,NULL);
}

CMsgQue::CMsgQue(int msg_id)
{
	m_msg_id = msg_id;
}

CMsgQue::~CMsgQue()
{

}




  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值