Linux IPC实践(7) --Posix消息队列
1. 创建/获取一个消息队列
#include <fcntl.h> /* For O_* constants */ #include <sys/stat.h> /* For mode constants */ #include <mqueue.h> mqd_t mq_open(const char *name, int oflag); //专用于打开一个消息队列 mqd_t mq_open(const char *name, int oflag, mode_t mode, struct mq_attr *attr);
参数:
name: 消息队列名字;
oflag: 与open函数类型, 可以是O_RDONLY, O_WRONLY, O_RDWR, 还可以按位或上O_CREAT, O_EXCL, O_NONBLOCK.
mode: 如果oflag指定了O_CREAT, 需要指定mode参数;
attr: 指定消息队列的属性;
返回值:
成功: 返回消息队列文件描述符;
失败: 返回-1;
注意-Posix IPC名字限制:
1. 必须以”/”开头, 并且后面不能还有”/”, 形如:/file-name;
2. 名字长度不能超过NAME_MAX
3. 链接时:Link with -lrt.
/** System V 消息队列
通过msgget来创建/打开消息队列
int msgget(key_t key, int msgflg);
**/
2. 关闭一个消息队列
int mq_close(mqd_t mqdes); /** System V 消息队列没有类似的该函数调用**/
3. 删除一个消息队列
int mq_unlink(const char *name); /** System V 消息队列 通过msgctl函数, 并将cmd指定为IPC_RMID来实现 int msgctl(int msqid, int cmd, struct msqid_ds *buf); **/
//示例 int main() { mqd_t mqid = mq_open("/abc", O_CREAT|O_RDONLY, 0666, NULL); if (mqid == -1) err_exit("mq_open error"); cout << "mq_open success" << endl; mq_close(mqid); mq_unlink("/abc"); cout << "unlink success" << endl; }
4. 获取/设置消息队列属性
int mq_getattr(mqd_t mqdes, struct mq_attr *attr); int mq_setattr(mqd_t mqdes, struct mq_attr *newattr, struct mq_attr *oldattr);
参数:
newattr: 需要设置的属性
oldattr: 原来的属性
//struct mq_attr结构体说明 struct mq_attr { long mq_flags; /* Flags: 0 or O_NONBLOCK */ long mq_maxmsg; /* Max. # of messages on queue: 消息队列能够保存的消息数 */ long mq_msgsize; /* Max. message size (bytes): 消息的最大长度 */ long mq_curmsgs; /* # of messages currently in queue: 消息队列当前保存的消息数 */ };
/** System V 消息队列
通过msgctl函数, 并将cmd指定为IPC_STAT/IPC_SET来实现
int msgctl(int msqid, int cmd, struct msqid_ds *buf);
**/
/** 示例: 获取消息队列的属性 **/ int main(int argc,char **argv) { mqd_t mqid = mq_open("/test", O_RDONLY|O_CREAT, 0666, NULL); if (mqid == -1) err_exit("mq_open error"); struct mq_attr attr; if (mq_getattr(mqid, &attr) == -1) err_exit("mq_getattr error"); cout << "Max messages on queue: " << attr.mq_maxmsg << endl; cout << "Max message size: " << attr.mq_msgsize << endl; cout << "current messages: " << attr.mq_curmsgs << endl; mq_close(mqid); return 0; }
5. 发送消息
int mq_send(mqd_t mqdes, const char *msg_ptr, size_t msg_len, unsigned msg_prio);
参数:
msg_ptr: 指向需要发送的消息的指针
msg_len: 消息长度
msg_prio: 消息的优先级
/** System V 消息队列
通过msgsnd函数来实现消息发送
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
**/
/** 示例: 向消息队列中发送消息, prio需要从命令行参数中读取 **/ struct Student { char name[36]; int age; }; int main(int argc,char **argv) { if (argc != 2) err_quit("./send <prio>"); mqd_t mqid = mq_open("/test", O_WRONLY|O_CREAT, 0666, NULL); if (mqid == -1) err_exit("mq_open error"); struct Student stu = {"xiaofang", 23}; unsigned prio = atoi(argv[1]); if (mq_send(mqid, (const char *)&stu, sizeof(stu), prio) == -1) err_exit("mq_send error"); mq_close(mqid); return 0; }
6. 从消息队列中读取消息
ssize_t mq_receive(mqd_t mqdes, char *msg_ptr, size_t msg_len, unsigned *msg_prio);
参数:
msg_len: 读取的消息的长度, 注意: 此值一定要等于mq_attr::mq_msgsize的值, 该值可以通过mq_getattr获取, 但一般是8192字节 [this must be greater than the mq_msgsize attribute of the queue (see mq_getattr(3)).]
msg_prio: 保存获取的消息的优先级
返回值:
成功: 返回读取的消息的字节数
失败: 返回-1
注意: 读取的永远是消息队列中优先级最高的最早的消息, 如果消息队列为, 如果不指定为非阻塞模式, 则mq_receive会阻塞;
/** System V 消息队列
通过msgrcv函数来实现消息发送的
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);
**/
/** 示例: 从消息队列中获取消息 **/ int main(int argc,char **argv) { mqd_t mqid = mq_open("/test", O_RDONLY); if (mqid == -1) err_exit("mq_open error"); struct Student buf; int nrcv; unsigned prio; struct mq_attr attr; if (mq_getattr(mqid, &attr) == -1) err_exit("mq_getattr error"); if ((nrcv = mq_receive(mqid, (char *)&buf, attr.mq_msgsize, &prio)) == -1) err_exit("mq_receive error"); cout << "receive " << nrcv << " bytes, priority: " << prio << ", name: " << buf.name << ", age: " << buf.age << endl; mq_close(mqid); return 0; }
7. 建立/删除消息到达通知事件
int mq_notify(mqd_t mqdes, const struct sigevent *sevp);
参数sevp:
NULL: 表示撤销已注册通知;
非空: 表示当消息到达且消息队列当前为空, 那么将得到通知;
通知方式:
1. 产生一个信号, 需要自己绑定
2. 创建一个线程, 执行指定的函数
注意: 这种注册的方式只是在消息队列从空到非空时才产生消息通知事件, 而且这种注册方式是一次性的!
//sigevent结构体 struct sigevent { int sigev_notify; /* Notification method */ int sigev_signo; /* Notification signal */ union sigval sigev_value; /* Data passed with notification */ void (*sigev_notify_function) (union sigval); /* Function used for thread notification (SIGEV_THREAD) */ void *sigev_notify_attributes; /* Attributes for notification thread (SIGEV_THREAD) */ pid_t sigev_notify_thread_id; /* ID of thread to signal (SIGEV_THREAD_ID) */ }; union sigval /* Data passed with notification */ { int sival_int; /* Integer value */ void *sival_ptr; /* Pointer value */ };
sigev_notify代表通知的方式: 一般常用两种取值:SIGEV_SIGNAL, 以信号方式通知; SIGEV_THREAD, 以线程方式通知
如果以信号方式通知: 则需要设定一下两个参数:
sigev_signo: 信号的代码
sigev_value: 信号的附加数据(实时信号)
如果以线程方式通知: 则需要设定以下两个参数:
sigev_notify_function
sigev_notify_attributes
/** Posix IPC所特有的功能, System V没有 **/
/**示例: 将下面程序多运行几遍, 尤其是当消息队列”从空->非空”, 多次”从空->非空”, 当消息队列不空时运行该程序时, 观察该程序的状态; **/ mqd_t mqid; long size; void sigHandlerForUSR1(int signo) { //将数据的读取转移到对信号SIGUSR1的响应函数中来 struct Student buf; int nrcv; unsigned prio; if ((nrcv = mq_receive(mqid, (char *)&buf, size, &prio)) == -1) err_exit("mq_receive error"); cout << "receive " << nrcv << " bytes, priority: " << prio << ", name: " << buf.name << ", age: " << buf.age << endl; } int main(int argc,char **argv) { // 安装信号响应函数 if (signal(SIGUSR1, sigHandlerForUSR1) == SIG_ERR) err_exit("signal error"); mqid = mq_open("/test", O_RDONLY); if (mqid == -1) err_exit("mq_open error"); // 获取消息的最大长度 struct mq_attr attr; if (mq_getattr(mqid, &attr) == -1) err_exit("mq_getattr error"); size = attr.mq_msgsize; // 注册消息到达通知事件 struct sigevent event; event.sigev_notify = SIGEV_SIGNAL; //指定以信号方式通知 event.sigev_signo = SIGUSR1; //指定以SIGUSR1通知 if (mq_notify(mqid, &event) == -1) err_exit("mq_notify error"); //死循环, 等待信号到来 while (true) pause(); mq_close(mqid); return 0; }
/** 示例:多次注册notify, 这样就能过多次接收消息, 但是还是不能从队列非空的时候进行接收, 将程序改造如下: **/ mqd_t mqid; long size; struct sigevent event; void sigHandlerForUSR1(int signo) { // 注意: 是在消息被读走之前进行注册, // 不然该程序就感应不到消息队列"从空->非空"的一个过程变化了 if (mq_notify(mqid, &event) == -1) err_exit("mq_notify error"); //将数据的读取转移到对信号SIGUSR1的响应函数中来 struct Student buf; int nrcv; unsigned prio; if ((nrcv = mq_receive(mqid, (char *)&buf, size, &prio)) == -1) err_exit("mq_receive error"); cout << "receive " << nrcv << " bytes, priority: " << prio << ", name: " << buf.name << ", age: " << buf.age << endl; } int main(int argc,char **argv) { // 安装信号响应函数 if (signal(SIGUSR1, sigHandlerForUSR1) == SIG_ERR) err_exit("signal error"); mqid = mq_open("/test", O_RDONLY); if (mqid == -1) err_exit("mq_open error"); // 获取消息的最大长度 struct mq_attr attr; if (mq_getattr(mqid, &attr) == -1) err_exit("mq_getattr error"); size = attr.mq_msgsize; // 注册消息到达通知事件 event.sigev_notify = SIGEV_SIGNAL; //指定以信号方式通知 event.sigev_signo = SIGUSR1; //指定以SIGUSR1通知 if (mq_notify(mqid, &event) == -1) err_exit("mq_notify error"); //死循环, 等待信号到来 while (true) pause(); mq_close(mqid); return 0; }
mq_notify 注意点总结:
1. 任何时刻只能有一个进程可以被注册为接收某个给定队列的通知;
2. 当有一个消息到达某个先前为空的队列, 而且已有一个进程被注册为接收该队列的通知时, 只有没有任何线程阻塞在该队列的mq_receive调用的前提下, 通知才会发出;
3. 当通知被发送给它的注册进程时, 该进程的注册被撤销. 进程必须再次调用mq_notify以重新注册(如果需要的话),但是要注意: 重新注册要放在从消息队列读出消息之前而不是之后(如同示例程序);
附-查看已经成功创建的Posix消息队列
#其存在与一个虚拟文件系统中, 需要将其挂载到系统中才能查看
Mounting the message queue filesystem On Linux, message queues are created in a virtual filesystem.
(Other implementations may also provide such a feature, but the details are likely to differ.) This
file system can be mounted (by the superuser, 注意是使用root用户才能成功) using the following commands:
mkdir /dev/mqueue
mount -t mqueue none /dev/mqueue
还可以使用cat查看该消息队列的状态, rm删除:
cat /dev/mqueue/abc
rm abc
还可umount该文件系统
umount /dev/mqueue
附-Makefile
.PHONY: clean all CC = g++ CPPFLAGS = -Wall -g BIN = main SOURCES = $(BIN.=.cpp) all: $(BIN) %.o: %.c $(CC) $(CPPFLAGS) -c $^ -o $@ main: main.o $(CC) $(CPPFLAGS) $^ -lrt -o $@ clean: -rm -rf $(BIN) *.o bin/ obj/ core
Linux IPC实践(7) --Posix消息队列的更多相关文章
- [转]Linux进程通信之POSIX消息队列
进程间的消息队列可以用这个实现,学习了下. http://blog.csdn.net/anonymalias/article/details/9799645?utm_source=tuicool&am ...
- linux c编程:Posix消息队列
Posix消息队列可以认为是一个消息链表. 有足够写权限的线程可以往队列中放置消息, 有足够读权限的线程可以从队列中取走消息 在某个进程往一个队列写入消息前, 并不需要另外某个进程在该队列上等待消息的 ...
- linux网络编程之posix消息队列
在前面已经学习了System v相关的IPC,今天起学习posix相关的IPC,关于这两者的内容区别,简单回顾一下: 而今天先学习posix的消息队列,下面开始: 接下来则编写程序来创建一个posix ...
- Linux IPC实践(10) --Posix共享内存
1. 创建/获取一个共享内存 #include <sys/mman.h> #include <sys/stat.h> /* For mode constants */ #inc ...
- POSIX 消息队列相关问题
一.查看和删除消息队列要想看到创建的posix消息队列,需要在root用户下执行以下操作:# mkdir /dev/mqueue# mount -t mqueue none /dev/mqueue删除 ...
- Linux进程间通信(IPC)编程实践(十二)Posix消息队列--基本API的使用
posix消息队列与system v消息队列的区别: (1)对posix消息队列的读总是返回最高优先级的最早消息,对system v消息队列的读则能够返回随意指定优先级的消息. (2)当往一个空队列放 ...
- Linux IPC实践(4) --System V消息队列(1)
消息队列概述 消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法(仅局限于本机); 每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值. 消息队列也有管道一样的不足: ...
- Linux IPC POSIX 消息队列
模型: #include<mqueue.h> #include <sys/stat.h> #include <fcntl.h> mq_open() //创建/获取消 ...
- Linux环境编程之IPC进程间通信(五):Posix消息队列1
对于管道和FIFO来说.必须应该先有读取者存在.否则先有写入者是没有意义的. 而消息队列则不同,它是一个消息链表,有足够写权限的线程可往别的队列中放置消息,有足够读权限的线程可从队列中取走消息.每一个 ...
随机推荐
- AleNet模型笔记
谁创造了AlexNet? AlexNet是有Hinton大神的弟子Alex Krizhevsky提出的深度卷积神经网络.它可视为LeNet的更深更宽的版本. AlexNet主要用到的技术 成功使用Re ...
- Luogu P1226 取余运算||快速幂_快速幂
超短代码 #include<iostream> #include<cstdio> using namespace std; long long b,p,k; long long ...
- js密码64加密
可以在客户端对密码进行简单的64位加密,服务端对应使用64位解密即可. /** * * Base64 encode / decode * * @author * @date * @email */ f ...
- Android自定义View(CustomCalendar-定制日历控件)
转载请标明出处: http://blog.csdn.net/xmxkf/article/details/54020386 本文出自:[openXu的博客] 目录: 1分析 2自定义属性 3onMeas ...
- 潜谈IT从业人员在传统IT和互联网之间的择业问题(下)-互联网公司
互联网带来的一片晴天 相对于传统行业来说,互联网行业要显得相对对技术人员尊重些. 在互联网行业中,采用的技术.概念也较传统形行业来说要新,技术人员也容易在此找到自己的一方净土. 因为互联网这个行当讲究 ...
- Objective-C方法与函数的区别
Objective-C方法与函数的区别 方法是唯对象所有 函数是不依赖于对象存在的 方法 函数 - (void)test; void test(); 方法是以减号 - 开头 - 类型要用()括起来 - ...
- 【Unity3D】 Unity Chan项目分享
写在前面 之前的一个博文里分享了日本Unity酱的项目,如果大家有去仔细搜Unity酱的话,就会发现日本Unity官方还放出了一个更完整的Unity酱的项目,感觉被萌化了!(事实上,Unity日本经常 ...
- Scala:函数和闭包
http://blog.csdn.net/pipisorry/article/details/52902271 Scala函数 Scala 有函数和方法,二者在语义上的区别很小.Scala 方法是类的 ...
- 查看oracle数据库的连接数以及用户 (转:http://blog.itpub.net/24558279/viewspace-752293/)
select2.查询oracle的并发连接数4select3.查看不同用户的连接数6select4.查看所有用户:8select5.查看用户或角色系统权限(直接赋值给用户或角色的系统权限):10sel ...
- nginx平台初识(二) 浏览器 HTTP 协议缓存机制详解
1.缓存的分类 缓存分为服务端侧(server side,比如 Nginx.Apache)和客户端侧(client side,比如 web browser). 服务端缓存又分为 代理服务器缓存 和 反 ...