进程间通信 --消息队列

一、基本知识

1. 概念

消息队列实际上是操作系统在内核为我们创建一个队列,多个进程可以通过向队列中添加节点或获取节点来进行数据传输,它是一个全双工通信,可读可写(可以发送数据,也可以接收数据)。

2. 如何获取数据

用户组织一个带有数据传输的数据块,添加到队列中,其他的进程从队列中获取数据块,也就是说消息队列传输的是一个个带有类型的数据块。

二、IPC(进程间通信)对象

1. 内核为每个IPC对象维护⼀个数据结构

struct ipc_perm {
    key_t __key;
    uid_t uid;  
    gid_t gid;
    uid_t cuid;
    gid_t cgid;
    unsigned short mode;
    unsigned short __seq;
};

key_t __key; 消息队列唯一的标识符

unsigned short mode;  权限

2. 消息队列结构

struct msqid_ds {
    struct ipc_perm msg_perm;
    struct msg *msg_first; 
    struct msg *msg_last;
    __kernel_time_t msg_stime;
    __kernel_time_t msg_rtime;
    __kernel_time_t msg_ctime; 
    unsigned long msg_lcbytes; 
    unsigned long msg_lqbytes;
    unsigned short msg_cbytes;
    unsigned short msg_qnum;
    unsigned short msg_qbytes;
    __kernel_ipc_pid_t msg_lspid;
    __kernel_ipc_pid_t msg_lrpid;
};

3. 操作系统中IPC相关命令

ipcs -q:查看消息队列

ipcs -m:查看共享内存

 ipcs -s:查看信号量

ipcrm -q/m/s msgid:删除IPC

三、流程

1. 创建消息队列或打开一个现有队

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgget(key_t key, int msgflg);
// 成功:消息队列ID 失败:-1

key:身份标识,某个消息队列的名字

msgflag:由九个权限标志构成,它们的⽤法和创建⽂件时使⽤的 mode 模式标志是⼀样的

#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok(const char *pathname, int proj_id);
// 成功:返回key_t值(即IPC 键值) 失败:-1,错误原因存于error中

ftok把一个已存在的路径名和一个整数标识符转换成一个key_t值,称为IPC键值(也称IPC key键值)

pathname:指定的文件,此文件必须存在且可存取

proj_id:计划代号(project ID)

2. msgsnd 将消息添加到队列尾端,msgrcv 从消息队列取数据

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
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);

msgid:由 msgget 函数返回的消息队列标识码

msgp:是⼀个指针,指针指向准备发送的消息

struct msgbuf {
    long mtype;
    char mtext[1024];
};

msgsz:是 msgp 指向的消息⻓度,这个⻓度不含保存消息类型的 long int ⻓整型

msgflg:控制着当前消息队列满或到达系统上限时将要发⽣的事情 msgflg=IPC_NOWAIT 表⽰队列满不等待,返回EAGAIN错误

msgtyp:第一个消息队列(等于0),消息类型为 msgtyp 的第一个消息队列(大于0),小于 msgtyp 绝对值的类型值最小的消息(小于0)

3. 释放消息队列:msgctl

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgctl(int msqid, int cmd, struct msqid_ds *buf);
// 成功:0 失败:-1

msgid:由msgget函数返回的消息队列标识码

cmd:IPC_STAT,IPC_SET,IPC_RMID

四、代码演示

Makefile

.PHONY:all
all:msg_q_s msg_q_c
msg_q_s:msg_q_s.c
    gcc msg_q_s.c -o msg_q_s 
msg_q_c:msg_q_c.c
    gcc msg_q_c.c -o msg_q_c 

.PHONY:clean
clean:
    rm -f msg_q_s msg_q_c

msg_q_s.c:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/msg.h>
#include <sys/ipc.h>

#define IPC_KEY 0x12345678
#define TYPE_S 1
#define TYPE_C 2

struct msgbuf {
    long mtype;
    char mtext[1024];
};

int main()
{
    int msg_id = -1;
    msg_id = msgget(IPC_KEY, IPC_CREAT | 0644);
    if (msg_id < 0)
    {
        perror("msgget error");
        return -1;
    }
    while (1)
    {
        /* 先接收,后发送 */
        struct msgbuf buf;
        msgrcv(msg_id, &buf, 1024, TYPE_C, 0);
        printf("client say:[%s]\n", buf.mtext);

        memset(&buf, 0x00, sizeof(struct msgbuf));
        buf.mtype = TYPE_S;
        scanf("%s", buf.mtext);
        msgsnd(msg_id, &buf, 1024, 0);
    }
    msgctl(msg_id, IPC_RMID, NULL);
    return 0;
}

msg_q_c.c:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/msg.h>
#include <sys/ipc.h>

#define IPC_KEY 0x12345678
#define TYPE_S 1
#define TYPE_C 2

struct msgbuf {
    long mtype;
    char mtext[1024];
};

int main()
{
    int msg_id = -1;
    msg_id = msgget(IPC_KEY, IPC_CREAT | 0644);
    if (msg_id < 0)
    {
        perror("msgget error");
        return -1;
    }
    while (1)
    {
        /* 先发送,后接收 */
        struct msgbuf buf;

        memset(&buf, 0x00, sizeof(struct msgbuf));
        buf.mtype = TYPE_C;
        scanf("%s", buf.mtext);
        msgsnd(msg_id, &buf, 1024, 0);

        memset(&buf, 0x00, sizeof(struct msgbuf));
        msgrcv(msg_id, &buf, 1024, TYPE_S, 0);
        printf("serve say:[%s]\n", buf.mtext);
    }
    msgctl(msg_id, IPC_RMID, NULL);
    return 0;
}

运行截图:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值