以前对于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
#include
#include
#include
#include
#include
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()
{
}