Linux IPC实践(4) --System V消息队列(1)
消息队列概述
消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法(仅局限于本机);
每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值.
消息队列也有管道一样的不足: (1)每个消息的最长字节数的上限(MSGMAX); (2)系统中消息队列的总条数也有一个上限(MSGMNI); (3)每个消息队列所能够保存的总字节数是有上限的(MSGMNB) .
查看系统限制
cat /proc/sys/kernel/msgmax #最大消息长度限制
cat /proc/sys/kernel/msgmnb #消息队列总的字节数
cat /proc/sys/kernel/msgmni #消息条目数
管道 vs. 消息队列
管道 |
消息 |
流管道 |
有边界 |
先进先出 |
可以后进先出 |
IPC对象数据结构
//内核为每个IPC对象维护一个数据结构 struct ipc_perm { key_t __key; /* Key supplied to msgget(2) */ uid_t uid; /* Effective UID of owner */ gid_t gid; /* Effective GID of owner */ uid_t cuid; /* Effective UID of creator */ gid_t cgid; /* Effective GID of creator */ unsigned short mode; /* Permissions */ unsigned short __seq; /* Sequence number */ };
//消息队列特有的结构 struct msqid_ds { struct ipc_perm msg_perm; /* Ownership and permissions 各类IPC对象所共有的数据结构*/ time_t msg_stime; /* Time of last msgsnd(2) */ time_t msg_rtime; /* Time of last msgrcv(2) */ time_t msg_ctime; /* Time of last change */ unsigned long __msg_cbytes; /* Current number of bytes in queue (nonstandard) 消息队列中当前所保存的字节数 */ msgqnum_t msg_qnum; /* Current number of messages in queue 消息队列中当前所保存的消息数 */ msglen_t msg_qbytes; /* Maximum number of bytes allowed in queue 消息队列所允许的最大字节数 */ pid_t msg_lspid; /* PID of last msgsnd(2) */ pid_t msg_lrpid; /* PID of last msgrcv(2) */ };
消息队列在内核中的表示
消息在消息队列中是以链表形式保存的, 每个节点的类型类似如下:
struct msq_Node { Type msq_type; //类型 Length msg_len; //长度 Data msg_data; //数据 struct msg_Node *next; };
消息队列API
#include <sys/types.h> #include <sys/ipc.h> #include <sys/msg.h> 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
功能:用来创建和访问一个消息队列
int msgget(key_t key, int msgflg);
参数:
key: 某个消息队列的名字
msgflg:由九个权限标志构成,如0644,它们的用法和创建文件时使用的mode模式标志是一样的(但是消息队列没有x(执行)权限)
返回值:
成功返回消息队列编号,即该消息队列的标识码;失败返回-1
msgget调用关系图
/** 示例1: 在msgflg处指定IPC_CREAT, 如果不存在该消息队列, 则创建之**/ int main(int argc, char *argv[]) { //指定IPC_CREAT,如果不存在, 则创建消息队列 int msgid = msgget(1234, 0666|IPC_CREAT); if (msgid == -1) err_exit("msgget error"); cout << "msgget success" << endl; }
/** 示例2:IPC_CREAT|IPC_EXCL, 如果该消息队列已经存在, 则返回出错 **/ int main(int argc, char *argv[]) { //指定IPC_EXCL, 如果已经存在,则报告文件已经存在(错误) int msgid = msgget(1234, 0666|IPC_CREAT|IPC_EXCL); if (msgid == -1) err_exit("msgget error"); cout << "msgget success" << endl; }
/**示例3:将key指定为IPC_PRIVATE(值为0) 将key指定为IPC_PRIVATE之后,则msgget就一定会创建一个新的消息队列, 而且每次创建的消息队列的描述符都是不同的! 因此, 除非将MessageID(key)传送给其他进程(除非有关联的进程),其他进程也无法使用该消息队列(血缘fork除外) 因此, IPC_PRIVATE创建的消息队列,只能用在与当前进程有关系的进程中使用! **/ int main(int argc, char *argv[]) { //指定IPC_PRIVATE int msgid = msgget(IPC_PRIVATE, 0666|IPC_CREAT|IPC_EXCL); if (msgid == -1) err_exit("msgget error"); cout << "msgget success" << endl; }
/** 示例4: 仅打开消息队列时, msgflg选项可以直接忽略(填0), 此时是以消息队列创建时的权限进行打开 **/ int main(int argc, char *argv[]) { int msgid = msgget(1234, 0); if (msgid == -1) err_exit("msgget error"); cout << "msgget success" << endl; cout << "msgid = " << msgid << endl; }
//示例5:低权限创建,高权限打开 int main() { //低权限创建 int msgid = msgget(0x255,0444 | IPC_CREAT); if (msgid < 0) err_exit("mesget error"); else cout << "Create Mes OK, msgid = " << msgid << endl; //高权限打开 msgid = msgget(0x255,0644 | IPC_CREAT); if (msgid < 0) err_exit("mesget error"); else cout << "Create Mes OK, msgid = " << msgid << endl; }
msgctl函数
功能:获取/设置消息队列的信息
int msgctl(int msqid, int cmd, struct msqid_ds *buf);
参数:
msqid: 由msgget函数返回的消息队列标识码
cmd:是将要采取的动作(见下)
cmd:将要采取的动作(有三个可取值),分别如下:
/** 示例1: IPC_RMID, 删除消息队列 注意: 消息队列并没有运用”引用计数”的功能 **/ int main() { int msgid = msgget(1234, 0); if (msgid == -1) err_exit("msgget error"); if (msgctl(msgid, IPC_RMID, NULL) == -1) err_exit("msgctl IPC_RMID error"); cout << "msgctl IPC_RMID success" << endl; }
/** 示例2: IPC_STAT **/ int main() { int msgid = msgget(0x255, 0666|IPC_CREAT); if (msgid == -1) err_exit("msgget error"); struct msqid_ds buf; if (msgctl(msgid,IPC_STAT,&buf) == -1) err_exit("msgctl error"); printf("buf.msg_perm.mode = %o\n",buf.msg_perm.mode); //%o以八进制打印 printf("buf.__key = %x\n", buf.msg_perm.__key); //%x以十六进制打印 cout << "buf.__msg_cbytes = " << buf.__msg_cbytes << endl; cout << "buf.msg_qbytes = " << buf.msg_qbytes << endl; cout << "buf.msg_lspid = " << buf.msg_lspid << endl; }
/** 实践:IPC_SET,一般需要先获取,然后再设置 **/ int main() { int msgid = msgget(0x255, 0); if (msgid == -1) err_exit("msgget error"); //获取消息队列的属性 struct msqid_ds buf; if (msgctl(msgid,IPC_STAT,&buf) == -1) err_exit("msgctl error"); //设置消息队列的属性 buf.msg_perm.mode = 0600; if (msgctl(msgid, IPC_SET, &buf) == -1) err_exit("msgctl error"); //获取并打印 bzero(&buf, sizeof(buf)); if (msgctl(msgid, IPC_STAT, &buf) == -1) err_exit("msgctl IPC_STAT error"); printf("mode = %o\n", buf.msg_perm.mode); }
附-查看系统中的IPC对象
ipcs
删除消息队列
ipcrm -q [msqid]
或 ipcrm -Q [key] #如果key不等于0的话
Linux IPC实践(4) --System V消息队列(1)的更多相关文章
- Linux IPC实践(6) --System V消息队列(3)
消息队列综合案例 消息队列实现回射客户/服务器 server进程接收时, 指定msgtyp为0, 从队首不断接收消息 server进程发送时, 将mtype指定为接收到的client进程的pid ...
- Linux IPC实践(5) --System V消息队列(2)
消息发送/接收API msgsnd函数 int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg); 参数 msgid: 由ms ...
- Linux进程通信之System V消息队列
System V消息队列是Open Group定义的XSI,不属于POSIX标准.System V IPC的历史相对很早,在上个世70年代后期有贝尔实验室的分支机构开发,80年代加入System V的 ...
- linux c编程:System V消息队列一
消息队列可以认为是一个消息链表,System V 消息队列使用消息队列标识符标识.具有足 够特权的任何进程都可以往一个队列放置一个消息,具有足够特权的任何进程都可以从一个给定队列读出一个消息.在某个进 ...
- linux网络编程之system v消息队列(二)
今天继续学习system v消息队列,主要是学习两个函数的使用,开始进入正题: 下面则开始用代码来使用一下该发送函数: 在运行之前,先查看一下1234消息队列是否已经创建: 用上次编写的查看消息队列状 ...
- Linux IPC实践(13) --System V IPC综合实践
实践:实现一个先进先出的共享内存shmfifo 使用消息队列即可实现消息的先进先出(FIFO), 但是使用共享内存实现消息的先进先出则更加快速; 我们首先完成C语言版本的shmfifo(基于过程调用) ...
- Linux IPC实践(11) --System V信号量(1)
信号量API #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> int semget ...
- Linux IPC实践(9) --System V共享内存
共享内存API #include <sys/ipc.h> #include <sys/shm.h> int shmget(key_t key, size_t size, int ...
- linux网络编程之system v消息队列(一)
经过上次对于进程通讯的一些理论的认识之后,接下来会通过实验来进一步加深对进程通讯的认识,话不多说,进入正题: 其实还可以通过管道,但是,管道是基于字节流的,所以通常会将它称为流管道,数据与数据之间是没 ...
随机推荐
- Python小代码_5_二维矩阵转置
使用列表推导式实现二维矩阵转置 matrix = [[1, 2, 3, 4], [5, 6, 7, 8], [9, 10, 11, 12]] print(matrix) matrix_t = [[ro ...
- 31. Next Permutation(中等,搞清楚啥是 next permutation)
Implement next permutation, which rearranges numbers into the lexicographically next greater permuta ...
- PHP使用prepare(),insert数据时要注意的一点!!!
今天看了PHP防SQL注入,使用预处理prepare,但是我insert数据时,总是插不进去,但是select却可以,弄了很久终于知道原来问题在这里,先上代码 <?php header('con ...
- solr服务器搭建
百度百科定义:Solr是一个独立的企业级搜索应用服务器,它对外提供类似于Web-service的API接口.用户可以通过http请求,向搜索引擎服务器提交一定格式的XML文件,生成索引:也可以通过Ht ...
- Vue nextTick 机制
背景 我们先来看一段Vue的执行代码: export default { data () { return { msg: 0 } }, mounted () { this.msg = 1 this.m ...
- substr函数用法详解
substr(string, start<,length>):从string的start位置开始提取字符串 length:要提取字符串的长度,若length为以下任意条件之一时,返回sta ...
- 安利三款提升幸福感的chrome插件
谷歌访问助手 chrome浏览器一直是各大码农推荐的比较好用的浏览器,速度快.插件多. 但是由于众所周知的原因导致了谷歌账号同步.扩展商店访问慢甚至打不开的情况. 谷歌访问助手就是用来解决此问题的. ...
- Java面试18|关于进程、线程与协程
1.IPC(Inter-Process Communication,进程间通信)与线程通信的几种方式 # 管道( pipe ):管道是一种半双工的通信方式,数据只能单向流动,而且只能在具有亲缘关系的进 ...
- Vue国际化处理 vue-i18n 以及项目自动切换中英文
1. 环境搭建 命令进入项目目录,执行以下命令安装vue 国际化插件vue-i18n npm install vue-i18n --save 2. 项目增加国际化翻译文件 在项目的src下添加lang ...
- Go 语言 goto 语句
Go 语言的 goto 语句可以无条件地转移到过程中指定的行. goto语句通常与条件语句配合使用.可用来实现条件转移, 构成循环,跳出循环体等功能. 但是,在结构化程序设计中一般不主张使用goto语 ...