Linux IPC实践(5) --System V消息队列(2)

消息发送/接收API

msgsnd函数

int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);

参数

msgid: 由msgget函数返回的消息队列标识码, 也可以是通过ipcs命令查询出来的一个已经存在的消息队列的ID号

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

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

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

消息结构在两方面受到制约: (1)它必须小于系统规定的上限值(MSGMAX); (2)它必须以一个long int长整数开始,接收者函数将利用这个长整数确定消息的类型;

//消息结构参考形式如下:
struct msgbuf
{
    long mtype;       /* message type, must be > 0 */
    char mtext[1];    /* message data, 可以设定为更多的字节数 */
};
/**示例1:
测试1: 发送消息的最大长度为8192字节, 一旦超过这个值, 则msgsnd出错, 提示 Invalid argument错误;
测试2: 消息队列所能够接收的最大字节数16384字节, 一旦超过这个长度, 如果msgflg为0(阻塞模式), 则进程会一直阻塞下去, 直到有进程来将消息取走; 而如果msgflg为IPC_NOWAIT模式, 则一个字节也不会写入消息队列, 直接出错返回;
**/
int main(int argc, char *argv[])
{
    if (argc != 3)
        err_quit("Usage: ./main <type> <length>");

    int type = atoi(argv[1]);
    int len = atoi(argv[2]);

    int msgid = msgget(0x255, 0666|IPC_CREAT);
    if (msgid == -1)
        err_exit("msgget error");

    struct msgbuf *buf;
    buf = (struct msgbuf *)malloc(len + sizeof(msgbuf::mtype));
    buf->mtype = type;
    if (msgsnd(msgid, buf, len, IPC_NOWAIT) == -1)
        err_exit("msgsnd error");
}

msgrcv函数

ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);

参数

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

msgp:是一个指针,指针指向准备接收的消息;

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

msgtype:它可以实现接收优先级的简单形式(见下图)

msgflg:控制着队列中没有相应类型的消息可供接收时将要发生的事(见下图)

返回值:

成功->返回实际放到接收缓冲区里去的字节数(注意: 此处并不包含msgbuf中的mtype的长度[man-page: msgrcv() returns the number of bytes actually copied into the mtext array.]);

失败->返回-1;


msgtyp参数


msgtyp=0


返回队列第一条信息


msgtyp>0


返回队列第一条类型等于msgtype的消息


msgtyp<0


返回队列第一条类型小于等于(<=)msgtype绝对值的消息,并且是满足条件的消息类型最小的消息(按照类型进行排序的顺序进行接收消息)


msgflg参数


msgflg=IPC_NOWAIT


队列没有可读消息不等待,返回ENOMSG错误。


msgflg=MSG_NOERROR


消息大小超过msgsz(msgrcv 函数的第三个参数)时被截断, 并且不会报错


msgtyp>0且msgflg=MSG_EXCEPT


接收类型不等于msgtype的第一条消息

/** 示例2: 消息接收(配合示例1中程序使用)
说明: 	-t [number], 指定接收消息的类型, 类型为number的值
-n ,指定以IPC_NOWAIT模式接收消息
**/
int main(int argc, char *argv[])
{
    /** 解析参数 **/
    int type = 0;
    int flag = 0;
    int opt;
    while ((opt = getopt(argc, argv, "nt:")) != -1)
    {
        switch (opt)
        {
        case ‘n‘:   // 指定IPC_NOWAIT选项
            flag |= IPC_NOWAIT;
            break;
        case ‘t‘:   // 指定接收的类型, 如果为0的话,说明是按照顺序接收
            type = atoi(optarg);
            break;
        default:
            exit(EXIT_FAILURE);
        }
    }

    int msgid = msgget(0x255, 0);
    if (msgid == -1)
        err_exit("msgget error");

    const int MSGMAX = 8192;    //指定一条消息的最大长度
    struct msgbuf *buf;
    buf = (struct msgbuf *)malloc(MSGMAX + sizeof(buf->mtype));

    ssize_t nrcv;
    if ((nrcv = msgrcv(msgid, buf, MSGMAX, type, flag)) == -1)
        err_exit("msgrcv error");
    cout << "recv " << nrcv << " bytes, type = " << buf->mtype << endl;
}

/** 综合示例: msgsnd/msgrcv, 消息发送/接收实践 **/
//1. 消息发送
int main()
{
    int msgid = msgget(0x1234,0666|IPC_CREAT);
    if (msgid == -1)
        err_exit("msgget error");

    struct msgBuf myBuffer;
    for (int i = 0; i < 128; ++i)
    {
        myBuffer.mtype = i+1;
        sprintf(myBuffer.mtext,"Hello, My number is %d",i+1);
        if (msgsnd(msgid,&myBuffer,strlen(myBuffer.mtext),IPC_NOWAIT) == -1)
            err_exit("msgsnd error");
    }
}
//2. 消息接收:从队首不断的取数据
int main(int argc, char *argv[])
{
    int msgid = msgget(0x1234, 0);
    if (msgid == -1)
        err_exit("msgget error");

    struct msgBuf buf;
    ssize_t nrcv;
    while ((nrcv = msgrcv(msgid, &buf, sizeof(buf.mtext), 0, IPC_NOWAIT)) > 0)
    {
        cout << "recv " << nrcv << " bytes, type: " << buf.mtype
        << ", message: " << buf.mtext << endl;
    }
}

[附]-getopt函数的用法

#include <unistd.h>
int getopt(int argc, char * const argv[],
                  const char *optstring);

extern char *optarg;
extern int optind, opterr, optopt;
//示例: 解析 ./main -n -t 3 中的参数选项
int main(int argc, char *argv[])
{
    while (true)
    {
        int opt = getopt(argc, argv, "nt:");
        if (opt == ‘?‘)
            exit(EXIT_FAILURE);
        else if (opt == -1)
            break;

        switch (opt)
        {
        case ‘n‘:
            cout << "-n" << endl;
            break;
        case ‘t‘:
            int n = atoi(optarg);
            cout << "-t " << n << endl;
            break;
        }
    }
}

时间: 2024-10-18 04:16:34

Linux IPC实践(5) --System V消息队列(2)的相关文章

Linux IPC实践(6) --System V消息队列(3)

消息队列综合案例 消息队列实现回射客户/服务器 server进程接收时, 指定msgtyp为0, 从队首不断接收消息 server进程发送时, 将mtype指定为接收到的client进程的pid client进程发送的时候, mtype指定为自己进程的pid client进程接收时, 需要将msgtyp指定为自己进程的pid, 只接收消息类型为自己pid的消息; // client/server进程接收/发送的数据结构 const int MSGMAX = 8192; struct msgBuf

Linux IPC实践(4) --System V消息队列(1)

消息队列概述 消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法(仅局限于本机); 每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值. 消息队列也有管道一样的不足: (1)每个消息的最长字节数的上限(MSGMAX); (2)系统中消息队列的总条数也有一个上限(MSGMNI); (3)每个消息队列所能够保存的总字节数是有上限的(MSGMNB) . 查看系统限制 cat /proc/sys/kernel/msgmax  #最大消息长度限制 cat /proc/sys

Linux IPC实践(13) --System V IPC综合实践

实践:实现一个先进先出的共享内存shmfifo 使用消息队列即可实现消息的先进先出(FIFO), 但是使用共享内存实现消息的先进先出则更加快速; 我们首先完成C语言版本的shmfifo(基于过程调用), 然后在此基础上实现C++版本的ShmFifo, 将1块共享内存与3个信号量(1个mutext信号量, 1个full信号量, 1个empty信号量)封装成一个类ShmFifo, 然后编写各自的测试代码; shmfifo说明: 将申请到的共享内存作为一块缓冲区, 将该内存的首部(p_shm到p_pa

Linux IPC实践(11) --System V信号量(1)

信号量API #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> int semget(key_t key, int nsems, int semflg); int semctl(int semid, int semnum, int cmd, ...); int semop(int semid, struct sembuf *sops, unsigned nsops); semget int

Linux IPC实践(9) --System V共享内存

共享内存API #include <sys/ipc.h> #include <sys/shm.h> int shmget(key_t key, size_t size, int shmflg); void *shmat(int shmid, const void *shmaddr, int shmflg); int shmdt(const void *shmaddr); int shmctl(int shmid, int cmd, struct shmid_ds *buf); //

Linux IPC实践(12) --System V信号量(2)

实践1:信号量实现进程互斥 父子进程执行流程如下: 父进程 子进程 P P O(print) X(print) sleep sleep O(print) X(print) V V sleep sleep 从图中可以看出, O或X总是成对出现的, 要么两个O, 要么两个X; /**P,V原语实现父子进程互斥使用终端**/ // 程序代码 int main(int argc,char *argv[]) { int semid = sem_create(IPC_PRIVATE); sem_setval

细说linux IPC(十):system V 消息队列

[版权声明:尊重原创,转载请保留出处:blog.csdn.net/shallnet 或 .../gentleliu,文章仅供学习交流,请勿用于商业用途] system V消息队列和posix消息队列类似,linux系统这两种消息队列都支持.先来看一下system V消息队列相关操作及其函数. msgget()函数创建一个消息队列或打开一个消息队列. #include <sys/types.h> #include <sys/ipc.h> #include <sys/msg.h&

System V消息队列(2)

msgsnd函数 功能:把一条消息添加到消息队列中 原型 int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg); 参数 msgid: 由msgget函数返回的消息队列标识码 msgp:是一个指针,指针指向准备发送的消息, msgsz:是msgp指向的消息长度,这个长度不含保存消息类型的那个long int长整型 msgflg:控制着当前消息队列满或到达系统上限时将要发生的事情 返回值:成功返回0:失败返回-1 msgs

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

一.介绍 在学习UNIX网络编程 卷1时,我们当时可以利用Socket套接字来实现回射客户/服务器程序,但是Socket编程是存在一些不足的,例如: 1. 服务器必须启动之时,客户端才能连上服务端,并与服务端进行通信: 2. 利用套接口描述符进行通信,必须知道对端的IP与端口. 二.相关函数介绍 下面,我们利用System V消息队列来实现进程间的通信: 首先,我们先来了解一下下面几个函数: 1. msgget: 该函数用于打开或创建消息队列,其作用相当与文件操作函数open. #include