day12(IO进程)进程间的通信---消息队列

消息队列 message queue

传统:无名管道、有名管道、信号

system V: 共享内存、信号灯集、消息队列

队列原则:

按消息的类型添加或读取消息

1. 特点:

  • 消息队列是IPC对象(活动在内核级别的一种进程间通信的工具)的一种
  • 一个消息队列由一个标识符 (即队列ID)来标识
  • 消息队列就是一个消息的列表。用户可以在消息队列中添加消息、读取消息等
  • 消息队列可以按照类型(自己设一个值作为类型)来发送/接收消息
    • 消息队列个数最多为16个;
    • 消息队列总容量最多为16384字节;
    • 每个消息内容最多为8192字节。

2. 步骤

  1. 创建key值: ftok
  2. 创建或打开消息队列: msgget
  3. 添加消息:按照消息的类型把消息添加到已经打开的消息队列末尾 msgsnd
  4. 读取消息:可以按照消息类型把消息从消息队列中取走 msgrcv
  5. 删除消息队列: msgctl

3. 操作命令

ipcs -q: 查看消息队列

ipcrm -q msgid: 删除消息队列

4. 函数接口

1.ftok

key_t ftok ( const char * pathname , int proj_id );

功能:创建出来的具有唯一映射关系的一个key值,帮助操作系统用来标识一块共享内存

参数:

Pathname:已经存在的可访问文件的名字

Proj_id:一个字符(因为只用低8位)

返回值:成功:key值

失败:-1

2.msgget

int msgget(key_t key, int flag);
功能:创建或打开一个消息队列
参数:  key值
       flag:创建消息队列的权限IPC_CREAT|IPC_EXCL|0666
返回值:成功:msgid
       失败:-1

3.msgsnd

int msgsnd ( int msqid , const void * msgp , size_t size , int flag );

功能:添加消息

参数:msqid:消息队列的ID

msgp:指向消息的指针。常用消息结构msgbuf如下:

struct msgbuf{

long mtype; //消息类型

char mtext[N]}//消息正文

size:发送的消息正文的字节数

flag:IPC_NOWAIT消息没有发送完成函数也会立即返回

0:直到发送完成函数才返回

返回值:成功:0

失败:-1

用法: msgsnd(msgid, &msg, sizeof(msg)-sizeof(long),0);

4.msgrcv

int msgrcv ( int msgid , void * msgp , size_t size , long msgtype , int flag );

功能:读取消息

参数:msgid:消息队列的ID

msgp:存放读取消息的空间

size:接受的消息正文的字节数(sizeof(msgp)-sizeof(long))

msgtype:

0:接收消息队列中第一个消息。

大于0:接收消息队列中第一个类型为msgtyp的消息.

小于0:接收消息队列中类型值不小于msgtyp的绝对值且类型值又最小的消息。

flag:

0:若无消息函数会一直阻塞

IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG

返回值:成功:接收到的消息的长度

失败:-1

5.msgctl

int msgctl ( int msgqid , int cmd , struct msqid_ds * buf );

功能:对消息队列的操作,删除消息队列

参数:msqid:消息队列的队列ID

cmd:

IPC_STAT:读取消息队列的属性,并将其保存在buf指向的缓冲区中。

IPC_SET:设置消息队列的属性。这个值取自buf参数。

IPC_RMID:从系统中删除消息队列。

buf:消息队列缓冲区

返回值:成功:0

失败:-1

用法:msgctl(msgid, IPC_RMID, NULL);

打开或创建消息队列:

操作消息队列:
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <errno.h>

struct msgbuf
{
    long type; //必须有,在第一个,表示消息的类型,值>0!
    int num;   //消息正文,自己定义
    char ch;
};

int main(int argc, char const *argv[])
{
    key_t key;
    int msgid;

    if ((key = ftok("msg.c", 'a')) < 0)
    {
        perror("ftok err");
        return -1;
    }
    printf("key: %#x\n", key);

    //打开或创建消息队列
    msgid = msgget(key, IPC_CREAT | IPC_EXCL | 0777);
    if (msgid <= 0)
    {
        if (errno == EEXIST)
            msgid = msgget(key, 0777); //如果已经存在消息队列那直接打开该消息队列
        else
        {
            perror("msgget err");
            return -1;
        }
    }
    printf("msgid: %d\n", msgid);

    //添加消息
    struct msgbuf msg;
    msg.type = 10;
    msg.num = 1000;
    msg.ch = 'a';
    msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0); //0:发完消息再返回,而不是立即返回函数

    msg.type = 20;
    msg.num = 2000;
    msg.ch = 'b';
    msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0);

    //读取消息
    struct msgbuf m;
    msgrcv(msgid, &m, sizeof(m) - sizeof(long), 20, 0); //0:阻塞,读完消息再返回

    printf("%d %c\n", m.num, m.ch);

    //删除消息队列
    msgctl(msgid, IPC_RMID, NULL);

    return 0;
}

进程间通信:

  • 17
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值