注意

在涉及到posix消息的函数时, gcc 编译时要加-lrt参数, 如

gcc -lrt unpipc.c mqpack.c send.c -o send

gcc -lrt unpipc.c mqpack.c receive.c -o receive

posix消息的name必须以/开头, 例

./send /msgname

基本函数

#include <mqueue.h>
mqd_t mq_open(const char *name,int oflag, ...
/* mode_t mode, struct mq_attr *attr */);
int mq_close(mqd_t mqdes);
int mq_unlink(const char *name);
int mq_getattr(mqd_t mqdes, struct mq_attr *attr);
int mq_setattr(mqd_t mqdes,const struct mq_attr *attr,struct mq_attr *oattr);
int mq_send(mqd_t mqdes, const char *ptr, size_t len, unsigned int prio);
ssize_t mq_receive(mqd_t mqdes, char *ptr, size_t len, unsigned int *priop);
int mq_notify(mqd_t mqdes, const struct sigevent *notification); #include <signal.h>
int sigwait(const sigset_t *set, int *sig);

posix消息队列的读写可以想像成文件的读写:

mq_close用于关闭mqd/关闭文件fd

mq_unlink用于删除消息队列/删除文件

mq_send发送消息到消息队列/写入文件

mq_receive从消息队列中取出消息/读取文件

mq_getattr获取消息属性/获取文件信息

mq_notify当队列中的消息数量由0到1时产生指定的信号/IO同步

mq_notify:

1.任意时候只有一个进程可以被注册为接收通知

2.当通知被发送到注册进程时,其注册即被撤销,就像老式的signal一样

3.当消息到达队列后再进行注册的话,不会产生消息

4.只有在消息队列的数量从0到1时才产生

sigwait:

1.sigwait是与当前进行的屏蔽信号函数sigprocmask搭配使用的

2.sigwait挂起线程,直到有感兴趣的信号到达,而这个信号也属于当前进程的屏蔽信号集

3.sigwait将信号从未决信号中删除,且不会调用信号处理函数,第二个参数返回该信号值

4.与sigsuspend不同的是,sigwait并不改变当前进程的信号屏蔽集,也不调用处理函数

包裹函数

mqpack.h

#ifndef _MQPACK_H
#define _MQPACK_H #include "unpipc.h"
#include <mqueue.h>
mqd_t Mq_open(const char *name,int oflag,mode_t mode,struct mq_attr *attr);
void Mq_close(mqd_t mqd);
void Mq_unlink(const char *name);
void Mq_getattr(mqd_t mqd,struct mq_attr *attr);
void Mq_setattr(mqd_t mqd,struct mq_attr *attr,struct mq_attr *oattr);
void Mq_send(mqd_t mqd,const char *ptr,size_t len,unsigned int prio);
ssize_t Mq_receive(mqd_t mqd,char *ptr,size_t len,unsigned int *priop);
void Mq_notify(mqd_t mqd,const struct sigevent *notification); #endif

mqpack.c

#include "mqpack.h"

mqd_t Mq_open(const char *name,int oflag,mode_t mode,struct mq_attr *attr){
mqd_t mqd=mq_open(name,oflag,mode,attr);
if(mqd == -1)
err_quit("mq_open error");
return(mqd);
}
void Mq_close(mqd_t mqd){
if(mq_close(mqd) == -1)
err_quit("mq_close error");
}
void Mq_unlink(const char *name){
if(mq_unlink(name) == -1)
err_quit("mq_unlink error");
}
void Mq_getattr(mqd_t mqd,struct mq_attr *attr){
if(mq_getattr(mqd,attr) == -1)
err_quit("mq_getattr error");
}
void Mq_setattr(mqd_t mqd,struct mq_attr *attr,struct mq_attr *oattr){
if(mq_setattr(mqd,attr,oattr) == -1)
err_quit("Mq_setattr error");
}
void Mq_send(mqd_t mqd,const char *ptr,size_t len,unsigned int prio){
if(mq_send(mqd,ptr,len,prio) == -1)
err_quit("mq_send error");
}
ssize_t Mq_receive(mqd_t mqd,char *ptr,size_t len,unsigned int *priop){
size_t n=mq_receive(mqd,ptr,len,priop);
if((-1 == n) && (errno != EAGAIN))
err_quit("mq_receive error");
return (n);
}
void Mq_notify(mqd_t mqd,const struct sigevent *notification){
if(mq_notify(mqd,notification) == -1)
err_quit("mq_notify error");
}

例子

编译和执行步骤就是文章开头的那三条命令

send.c

#include "unpipc.h"
#include "mqpack.h" int main(int argc,char *argv[]){
if(argc != 2)
err_quit("usage: /send <name>");
char buff[1024];
mqd_t mqd;
struct mq_attr attr;
mqd=Mq_open(argv[1],O_WRONLY|O_CREAT,0644,NULL); if(Fork() == 0){
if(execl("./receive","./receive",argv[1],(char *)0) == -1)
err_quit("execl error");
} sleep(2); while(1){
Fgets(buff,1024,stdin);
if(buff[strlen(buff)-1] == '\n') //去掉buff中的换行符
buff[strlen(buff)-1]='\0';
Mq_send(mqd,buff,strlen(buff),0);
}
Mq_close(mqd);
exit(0);
}

receive.c

#include "unpipc.h"
#include "mqpack.h" void print_info(mqd_t mqd,struct mq_attr *attr){
if(mq_getattr(mqd,attr) == -1)
err_quit("mq_getn error");
printf("max num of maxmsg = %ld,max size of msg = %ld\n",attr->mq_maxmsg,attr->mq_msgsize);
printf("current mum of maxmsg = %ld\n",attr->mq_curmsgs);
} int main(int argc,char *argv[]){
int signo;
mqd_t mqd;
void *buff;
ssize_t n;
sigset_t newmask;
struct mq_attr attr;
struct sigevent sigev; if(argc != 2)
err_quit("usage mqtest2 <name>");
mqd=Mq_open(argv[1],O_RDONLY|O_NONBLOCK,0,NULL);
Mq_getattr(mqd,&attr);
buff=Malloc(attr.mq_msgsize);
print_info(mqd,&attr); Sigemptyset(&newmask);
Sigaddset(&newmask,SIGUSR1);
Sigprocmask(SIG_BLOCK,&newmask,NULL); sigev.sigev_notify=SIGEV_SIGNAL;
sigev.sigev_signo=SIGUSR1;
Mq_notify(mqd,&sigev); for(;;){
Sigwait(&newmask,&signo);
if(signo == SIGUSR1){
Mq_notify(mqd,&sigev);
while((n=Mq_receive(mqd,buff,attr.mq_msgsize,NULL))>0)
printf("read %ld bytes\n",(long)n);
//if(errno != EAGAIN)
// err_quit("mq_receive error");
}
}
}

posix 消息队列的更多相关文章

  1. Linux IPC POSIX 消息队列

    模型: #include<mqueue.h> #include <sys/stat.h> #include <fcntl.h> mq_open() //创建/获取消 ...

  2. Posix消息队列

    转载于:http://blog.csdn.net/zx714311728/article/details/53197196 1.消息队列 消息队列可以认为是一个消息链表,消息队列是随内核持续的.队列中 ...

  3. 第5章 Posix 消息队列

    5.1 概述 消息队列可以认为是一个链表.有写权限的线程可往消息队列中放置消息,有读权限的线程可以从消息队列中取走消息. 消息队列和管道/FIFO的区别: (1)消息队列往一个队列中写消息前,并不需要 ...

  4. Posix消息队列实现机制

    本文是对<Unix 网络编程 卷2:进程通信>的笔记. 引言 消息队列是进程间通信的一种方式,可是如果不理解他的实现原理,会有众多不理解之处,下面就结合本书中的例子,对posix消息队列来 ...

  5. POSIX 消息队列相关问题

    一.查看和删除消息队列要想看到创建的posix消息队列,需要在root用户下执行以下操作:# mkdir /dev/mqueue# mount -t mqueue none /dev/mqueue删除 ...

  6. Linux IPC实践(7) --Posix消息队列

    1. 创建/获取一个消息队列 #include <fcntl.h> /* For O_* constants */ #include <sys/stat.h> /* For m ...

  7. 进程间通信--POSIX消息队列

    相关函数: mqd_t mq_open(const char *name, int oflag); mqd_t mq_send(mqd_t mqdes, const char *msg_ptr, si ...

  8. POSIX 消息队列 之 概述 链接方式

    NAMEmq_overview —— POSIX消息队列概述 DESCRIPTIONPOSIX消息队列允许进程以消息的形式交换数据.此API与System V消息队列(msgget(2),msgsnd ...

  9. Unix IPC之Posix消息队列(1)

    部分参考:http://www.cnblogs.com/Anker/archive/2013/01/04/2843832.html IPC对象的持续性:http://book.51cto.com/ar ...

随机推荐

  1. leetcode 题解:Binary Tree Level Order Traversal (二叉树的层序遍历)

    题目: Given a binary tree, return the level order traversal of its nodes' values. (ie, from left to ri ...

  2. web前端开发(1)

    web标准 结构标准.样式标准.行为标准分离 结构标准:xml xhtml  html 样式标准:css 行为标准:DOM  ECMAScript 符合标准的网页:1.标签小写  2.属性加引号 3. ...

  3. javascript中Math ceil(),floor(),round()三个函数的对比

    Math.ceil()执行的是向上舍入 Math.floor()执行向下舍入 Math.round()执行标准舍入 一下是一些补充: ceil():将小数部分一律向整数部分进位. 如: Math.ce ...

  4. 第四节 数据格式化和ModelAttribute注解的介绍

    从来都不坦荡,情绪都写在脸上:不开心的时候,不爱说话,笑也勉强. 课堂笔记,如果这么写,不仅仅是手速,还要有语速, 这样不太适合! --胖先生 关于数据传递: 客户端传递数据到服务端: 1.使用普通的 ...

  5. 【转】Oracle 中的 TO_DATE 和 TO_CHAR 函数 日期处理

    Oracle 中的 TO_DATE 和 TO_CHAR 函数oracle 中 TO_DATE 函数的时间格式,以 2008-09-10 23:45:56 为例 格式 说明 显示值 备注 Year(年) ...

  6. 手机app(功能)测试重点

    在手机客户端进行查看的测试重点:1.“点击加载更多”的分页处理技术,是否有重复的数据,数据显示是否完整,到达最后一页后是否还有数据进行显示2.数据的排序方式2.界面跳转是否正确3.出现异常情况是否有提 ...

  7. document.execCommand()命令小计

    2D-Position 允许通过拖曳移动绝对定位的对象. AbsolutePosition 设定元素的 position 属性为“absolute”(绝对). BackColor 设置或获取当前选中区 ...

  8. AngularJS尝鲜一

    第一个小例子,体验一下: <!DOCTYPE html> <html> <head> <title>Index</title> </h ...

  9. JQuery处理json与ajax返回JSON实例

    一.JSON的一些基础知识. JSON中对象通过“{}”来标识,一个“{}”代表一个对象,如{“AreaId”:”123”},对象的值是键值对的形式(key:value). “[]”,标识数组,数组内 ...

  10. swift创建对象use of undeclared type 自己的类

    在swift项目中,引用自己创建的类,编译会成功的,但是会出现红色感叹号,类似报错,如:swift创建对象use of  undeclared type 自己的类,或者 use of unresolv ...