linux消息队列简析

消息队列提供了一种在两个进程之间(相同进程或不同进程)传递数据的简单且有效的方法,使用者不需要关心锁的问题。



函数介绍:

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

此函数可生成供msgget,semget,shmget使用的键值。pathname和proj_id相同,则生成的key值相同


int msgget(key_t key, int msgflg);

创建或获取消息队列id


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);

消息队列接收函数


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

控制消息队列


1. 查看消息队列限制

$ipcs -l

------ Messages Limits --------
max queues system wide = 988 -->最多创建消息队列数
max size of message (bytes) = 8192 -->默认单个消息最大字节数
default max size of queue (bytes) = 16384 (16K)-->默认最多存储字节数


2. 查看当前系统消息队列信息

$ipcs -u

------ Messages Status --------
allocated queues = 0
used headers = 0
used space = 0 bytes


示例代码:

msg_queue_snd.c:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

#include "msg_queue.h"

int msg_id = -1;

int msg_queue_limit_test() //性能测试函数
{
    struct msqid_ds msq_ds;
    int ret = -1;
    int i = 0;
    int limit = 80;
    char buf[512] = "hello";
    msg_event_t msg;

    memset(&msq_ds, 0, sizeof(msq_ds));
    ret = msgctl(msg_id, IPC_STAT, &msq_ds);
    if (ret < 0)
    {
        printf("%s: msg get failed: %s\n", __FUNCTION__, strerror(errno));
        exit(1);
    }

    printf("msg_stime: %ld, msg_rtime: %ld, msg_ctime: %ld, msg_qnum: %ld,"
            "msg_qbytes:%ld\n",
            msq_ds.msg_stime, msq_ds.msg_rtime, msq_ds.msg_ctime, msq_ds.msg_qnum,
            msq_ds.msg_qbytes);

    memset(&msq_ds, 0, sizeof(msq_ds));
    msq_ds.msg_qbytes = 16384 * 2;

    ret = msgctl(msg_id, IPC_SET, &msq_ds);
    if (ret < 0)
    {
        printf("%s: msg set failed: %s\n", __FUNCTION__, strerror(errno));
        exit(1);
    }

    ret = msgctl(msg_id, IPC_STAT, &msq_ds);
    if (ret < 0)
    {
        printf("%s: msg get failed: %s\n", __FUNCTION__, strerror(errno));
        exit(1);
    }

    printf("2:msg_stime: %ld, msg_rtime: %ld, msg_ctime: %ld, msg_qnum: %ld,"
        "msg_qbytes:%ld\n",
        msq_ds.msg_stime, msq_ds.msg_rtime, msq_ds.msg_ctime, msq_ds.msg_qnum,
        msq_ds.msg_qbytes);

    msg.msg_type = 1;
    strcpy(msg.buf, buf);
    for (i = 0; i < limit; i++)
    {
        printf("i: %d\n", i);
        ret = msgsnd(msg_id, &msg, sizeof(msg) - sizeof(long int), IPC_NOWAIT);
        if (ret < 0)
        {
            printf("%s: msg send failed: %s\n", __FUNCTION__, strerror(errno));
            exit(1);
        }
    }
    return 0;
}

int main(int argc, char *argv[])
{
    int i = 0;
    key_t key = -1;
    char buf[32] = {0};
    msg_event_t msg;
    int ret = -1;
    struct msqid_ds msq_ds;

    for (i = 0; i < MSG_QUEUE_NUM; i++)
    {
        key = ftok(KEY_DIR, 0);
        printf("%s: key[%d]: %d\n", __FUNCTION__, i, key);
    }

    msg_id = msgget(key, IPC_CREAT | 0666);
    if (msg_id < 0)
    {
        printf("%s: msg get failed: %s\n", __FUNCTION__, strerror(errno));
        exit(1);
    }

    printf("%s: msg id:%d\n", __FUNCTION__, msg_id);

    memset(&msq_ds, 0, sizeof(msq_ds));
    ret = msgctl(msg_id, IPC_STAT, &msq_ds);
    if (ret < 0)
    {
        printf("%s: msg get failed: %s\n", __FUNCTION__, strerror(errno));
        exit(1);
    }

    printf("msg_stime: %ld, msg_rtime: %ld, msg_ctime: %ld, msg_qnum: %ld,"
            "msg_qbytes:%ld\n",
            msq_ds.msg_stime, msq_ds.msg_rtime, msq_ds.msg_ctime, msq_ds.msg_qnum,
            msq_ds.msg_qbytes);

    //msg_queue_limit_test();

    while (1)
    {
        memset(&msg, 0, sizeof(msg));
        ret = read(STDIN_FILENO, buf, sizeof(buf));
        if (ret <= 0)
        {
            printf("%s: read from stdin %s\n", __FUNCTION__, strerror(errno));
            break;
        }

        buf[ret - 1] = '\0';

        msg.msg_type = 1;
        strcpy(msg.buf, buf);
        ret = msgsnd(msg_id, &msg, sizeof(msg) - sizeof(long int), IPC_NOWAIT);
        if (ret < 0)
        {
            printf("%s: msg send failed: %s\n", __FUNCTION__, strerror(errno));
            exit(1);
        }

        if (0 == strcmp(buf, "exit"))
        {
            printf("msg snd exit\n");
            break;
        }
    } 

    return 0;
}

msg_queue_recv.c:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

#include "msg_queue.h"

int msg_id = -1;

int main(int argc, char *argv[])
{
    int i = 0;
    key_t key = -1;
    msg_event_t msg;
    int ret = -1;

    for (i = 0; i < MSG_QUEUE_NUM; i++)
    {
        key = ftok(KEY_DIR, 0);
        printf("%s: key[%d]: %d\n", __FUNCTION__, i, key);
    }

    msg_id = msgget(key, 0666);
    if (msg_id < 0)
    {
        printf("%s: msg get failed: %s\n", __FUNCTION__, strerror(errno));
        exit(1);
    }

    printf("%s: msg id:%d\n", __FUNCTION__, msg_id);

    while (1)
    {
        memset(&msg, 0, sizeof(msg));

        msg.msg_type = 0;

        ret = msgrcv(msg_id, &msg, sizeof(msg), 0, 0);
        if (ret < 0)
        {
            printf("%s: msg get failed: %s\n", __FUNCTION__, strerror(errno));
            exit(1);
        }
        printf("get msg: %s\n", msg.buf);
        if (0 == strcmp(msg.buf, "exit"))
        {
            printf("msg get exit\n");
            break;
        }
    }

    msgctl(msg_id, IPC_RMID, NULL);

    return 0;
}


msg_queue.h:
#ifndef __MSG_QUEUE_H__
#define __MSG_QUEUE_H__

#define KEY_DIR "/tmp"
#define MSG_QUEUE_NUM 1

typedef struct msg_event_s
{
    long int msg_type;

    char buf[512];
} msg_event_t;

#endif

Makefile:
all:
    gcc -o msg_snd msg_queue_snd.c -g
    gcc -o msg_recv msg_queue_recv.c -g
clean:
    rm msg_snd
    rm msg_recv

测试结果

1. 默认最多存储16384字节,以单个消息512字节计算,可以存储32个消息

msg_stime: 1491099133, msg_rtime: 1491099143, msg_ctime: 1491094456, msg_qnum: 0,msg_qbytes:16384
i: 0
i: 1
i: 2
i: 3
i: 4
i: 5
i: 6
i: 7
i: 8
i: 9
i: 10
i: 11
i: 12
i: 13
i: 14
i: 15
i: 16
i: 17
i: 18
i: 19
i: 20
i: 21
i: 22
i: 23
i: 24
i: 25
i: 26
i: 27
i: 28
i: 29
i: 30
i: 31
i: 32
msg_queue_limit_test: msg send failed: Resource temporarily unavailable


2.更改最大存储字节数为16384*2,以单个消息512字节计算,可以存储64个消息

msg_stime: 1491100080, msg_rtime: 1491100132, msg_ctime: 1491100080, msg_qnum: 0,msg_qbytes:32768
2:msg_stime: 1491100080, msg_rtime: 1491100132, msg_ctime: 1491100140, msg_qnum: 0,msg_qbytes:32768
i: 0
i: 1
i: 2
i: 3
i: 4
i: 5
i: 6
i: 7
i: 8
i: 9
i: 10
i: 11
i: 12
i: 13
i: 14
i: 15
i: 16
i: 17
i: 18
i: 19
i: 20
i: 21
i: 22
i: 23
i: 24
i: 25
i: 26
i: 27
i: 28
i: 29
i: 30
i: 31
i: 32
i: 33
i: 34
i: 35
i: 36
i: 37
i: 38
i: 39
i: 40
i: 41
i: 42
i: 43
i: 44
i: 45
i: 46
i: 47
i: 48
i: 49
i: 50
i: 51
i: 52
i: 53
i: 54
i: 55
i: 56
i: 57
i: 58
i: 59
i: 60
i: 61
i: 62
i: 63
i: 64
msg_queue_limit_test: msg send failed: Resource temporarily unavailable-->队列已满


3. 发送、接收测试

 $./msg_snd 
main: key[0]: 65538
main: msg id:98304
msg_stime: 1491100739, msg_rtime: 1491100747, msg_ctime: 1491100737, msg_qnum: 0,msg_qbytes:16384
hello
exit
msg snd exit


$./msg_recv 
main: key[0]: 65538
main: msg id:98304
get msg: hello
get msg: exit
msg get exit






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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

浪游东戴河

你就是这个世界的唯一

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值