IPC对象的持续性

  • 随进程持续 :一直存在直到打开的最后一个进程结束。(如pipe和FIFO)
  • 随内核持续 :一直存在直到内核自举(内核自举就是把主引导记录加载到内存,并跳转执行这段内存)或显示删除(如System V消息队列、共享内存、信号量)
  • 随文件系统持续 :一直存在直到显示删除,即使内核自举还存在。(POSIX消息队列、共享内存、信号量如果是使用映射文件来实现)

消息队列

  • 消息队列提供了一种从一个进程向另外一个进程发送一块数据的方法
  • 每个数据块都被认为是有一个类型,接受者进程接受的数据块可以有不同的类型值
    • 管道所传输的数据是基于字节流的,数据与数据之间是没有边界的,所以通常称管道为流管道;接受数据的时候,遵守先进先出的原则。
    • 消息队列传输的数据块是有类型的,通常把成为消息,消息与消息之间是有边界的;接受数据的时候,不一定要按照顺序的方式来接受(不一定要先进先出)
  • 消息队列也有管道一样的不足,就是每个消息的最大长度是有限的(MSGMAX),每个消息队列的总的字节数是有限的(MSGMNB),系统上消息队列的总数也是有一个上限(MSGMNI)

IPC对象数据结构

  内核为每个IPC对象维护一个数据结构

struct ipc_perm {
uid_t cuid; /* creator user ID */
gid_t cgid; /* creator group ID */
uid_t uid; /* owner user ID */
gid_t gid; /* owner group ID */
unsigned short mode; /* r/w permissions */
};

消息队列结构

struct msqid_ds {
struct ipc_perm msg_perm;
msgqnum_t msg_qnum; /* no of messages on queue */
msglen_t msg_qbytes; /* bytes max on a queue */
pid_t msg_lspid; /* PID of last msgsnd(2) call */
pid_t msg_lrpid; /* PID of last msgrcv(2) call */
time_t msg_stime; /* last msgsnd(2) time */
time_t msg_rtime; /* last msgrcv(2) time */
time_t msg_ctime; /* last change time */
};

消息队列在内核中的表示

消息队列函数

msgget

头文件
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> int msgget(key_t key, int msgflg);
作用:
创建或打开一个消息队列
参数:
key : 某个消息队列的名字
msgflg : 由9个权限标志构成,它们的用法和创建文件时使用的模式标志是一样的
返回值:
成功 : 返回一个非负整数,即消息队列的标识码
失败 : -1

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> #define ERR_EXIT(m) \
do \
{ \
perror(m); \
exit(EXIT_FAILURE); \
} while (0) int main(int argc, char const *argv[])
{ int msgid;
// 如果存在就打开,不存在就创建
// msgid = msgget(1234,0666 | IPC_CREAT); // 如果不存在就创建,存在打开就报错
// msgid = msgget(1234,0666 | IPC_CREAT | IPC_EXCL); //IPC_PRIVATE实际上等于0,当key为IPC_PRIVATE时,会创建一个新的消息队列,不能被其他进程共享;父子进程、兄弟进程可以共享,因为可以获取到msqid
// msgid = msgget(IPC_PRIVATE,0666 | IPC_CREAT | IPC_EXCL); //指定了IPC_PRIVATE后,可以不加IPC_CREAT | IPC_EXCL
// msgid = msgget(IPC_PRIVATE,0666); //如果消息队列已存在,可以直接打开
msgid = msgget(1234,0);
if(msgid == -1)
ERR_EXIT("msgget"); printf("msget success, msgid=%d\n!",msgid); return 0;
}

msgctl

int msgctl(int msqid, int cmd, struct msqid_ds *buf);
作用:
控制消息队列
参数:
msqid : 由msgget函数返回的消息队列标识码
cmd : 将要采取的动作
IPC_STAT : 把msgid_ds结构中的数据设置为消息队列的当前关联值
IPC_SET : 在进程有足够权限的前提下,把消息队列的当前关联值设置为msgid_ds数据结构中给出的值
IPC_RMID : 删除消息队列
buf : 返回值:
成功 : 0
失败 : -1

IPC_RMID

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>
#include <stdio.h> int main(int argc, char const *argv[])
{
int msqid;
msqid = msgget(1234, 0);
if(msqid == -1)
{
perror("msgget error");
exit(EXIT_FAILURE);
} printf("msqid : %d\n",msqid); msgctl(msqid,IPC_RMID,NULL); return 0;
}

IPC_STAT

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>
#include <stdio.h> int main(int argc, char const *argv[])
{
int msqid;
msqid = msgget(1234, 0);
if(msqid == -1)
{
perror("msgget error");
exit(EXIT_FAILURE);
} printf("msqid : %d\n",msqid); struct msqid_ds buf;
msgctl(msqid,IPC_STAT,&buf); printf("mode : %o\n",buf.msg_perm.mode);
printf("bytes : %ld\n",buf.__msg_cbytes);
printf("number : %lu\n",buf.msg_qnum);
printf("msgmnb : %lu\n",buf.msg_qbytes);
return 0;
}

IPC_SET

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>
#include <stdio.h> int main(int argc, char const *argv[])
{
int msqid;
msqid = msgget(1234, 0);
if(msqid == -1)
{
perror("msgget error");
exit(EXIT_FAILURE);
} printf("msqid : %d\n",msqid); struct msqid_ds buf;
msgctl(msqid,IPC_STAT,&buf);
sscanf("600", "%o", (unsigned int*)&buf.msg_perm.mode);
msgctl(msqid, IPC_SET, &buf); printf("mode : %o\n",buf.msg_perm.mode);
return 0;
}

其它

ipcs - 查看当前已打开的消息队列

ipcrm -S|-s key|semid - 删除打开的消息队列

ipcrm -q msqid - 删除打开的消息队列

ipcrm -Q 键 - 删除打开的消息队列(只能用于键不为0的)

第二十五章 system v消息队列(一)的更多相关文章

  1. 第二十六章 system v消息队列(二)

    msgsnd int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg); 作用: 把一条消息添加到消息队列中 参数: msqi ...

  2. 第6章 System V消息队列

    6.1 概述 System V消息队列在内核中是list存放的,头结点中有2个指针msg_first 和msg_last.其中每个节点包含:下个节点地址的指针.类型.长度.数据等. 6.2 函数 6. ...

  3. 第二十七章 system v消息队列(三)

    消息队列实现回射客户/服务器 msg_srv.c #include <stdio.h> #include <stdlib.h> #include <unistd.h> ...

  4. 第二十九章 System V共享内存

    共享内存数据结构 共享内存函数 shmget int shmget(key_t key, size_t size, int shmflg); 功能: 用于创建共享内存 参数: key : 这个共享内存 ...

  5. 进程间通信 System V 消息队列

    1.msgget (key_t ket,int flag) ; //创建一个新的消息队列或者访问一个已存在的消息队列 2.msgsnd(int msid, const void *ptr ,size_ ...

  6. Linux进程通信之System V消息队列

    System V消息队列是Open Group定义的XSI,不属于POSIX标准.System V IPC的历史相对很早,在上个世70年代后期有贝尔实验室的分支机构开发,80年代加入System V的 ...

  7. 利用System V消息队列实现回射客户/服务器

    一.介绍 在学习UNIX网络编程 卷1时,我们当时可以利用Socket套接字来实现回射客户/服务器程序,但是Socket编程是存在一些不足的,例如: 1. 服务器必须启动之时,客户端才能连上服务端,并 ...

  8. UNIX环境高级编程——system V消息队列

    unix早期通信机制中的信号能够传送的信息量有限,管道则只能传送无格式字节流,这远远是不够的.     消息队列(也叫报文队列)客服了这些缺点:     消息队列就是一个消息的链表.     可以把消 ...

  9. Gradle 1.12用户指南翻译——第二十五章. Scala 插件

    其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://g ...

随机推荐

  1. react redux 二次开发流程

    在一个大项目中如何引入redux及其相关技术栈(react-redux redux-thunk redux-immutable ),已经成为react前端工程师不可或缺的技能,下面通过实现一个简单的t ...

  2. CMD的最佳“代替品”

    让CMD成为历史 Windows用户大多都使用过"cmd",cmd被称为"阉割版"的DOS系统~ 很多用户除此之外,还喜欢Linux命令行~但是CMD的命令和L ...

  3. windows下 python 如何安装pygame模块

    本机系统:win7,Pyhon版本: 3.6.0 1. 安装下载python官网 https://www.python.org/ 下载地址 https://www.python.org/downloa ...

  4. Spring Boot (九): 微服务应用监控 Spring Boot Actuator 详解

    1. 引言 在当前的微服务架构方式下,我们会有很多的服务部署在不同的机器上,相互是通过服务调用的方式进行交互,一个完整的业务流程中间会经过很多个微服务的处理和传递,那么,如何能知道每个服务的健康状况就 ...

  5. Zookeeper监控(Zabbix)

      一直在弄监控,这些个中间件Zookeeper.Kafka......,平时也只知道一点皮毛,也就搭建部署过,没有真正的用过,一般都是大数据的同学在用,作为运维人员我需要对他做一个监控,由于对他不是 ...

  6. MongoDB 学习笔记之 索引选项和重建索引

    索引选项: {background:true}在后台创建索引,索引在构建过程中,其他客户端仍然可以查询数据,不会阻塞. db.comments.createIndex({anonymous: 1},{ ...

  7. Java 并发编程(三):如何保证共享变量的可见性?

    上一篇,我们谈了谈如何通过同步来保证共享变量的原子性(一个操作或者多个操作要么全部执行并且执行的过程不会被任何因素打断,要么就都不执行),本篇我们来谈一谈如何保证共享变量的可见性(多个线程访问同一个变 ...

  8. INTELLIJ MAC查看类结构快捷键

    mac下intellij查看类结构快捷键有两种形式. 方法一 alt+7,通过窗口展示类结果,点击对应的方法,类中跳转到对应的位置,但此窗口并不会消失.如下图: 方法二 默认使用command+F12 ...

  9. 即学即用的 30 段 Python 实用代码

    [☞ 分享:最全最新的Python学习大礼包 ☜ 点击查看](https://mp.weixin.qq.com/s?__biz=MzU2MzgyODA4OA==&mid=100000592&a ...

  10. 全面系统Python3入门+进阶课程 ✌✌

    全面系统Python3入门+进阶课程 (一个人学习或许会很枯燥,但是寻找更多志同道合的朋友一起,学习将会变得更加有意义✌✌) 无论是大数据.人工智能还是机器学习,Python都是最热门的首选语言 ,这 ...