Linux系统编程-进程间通信(消息队列)

简介: 前面文章介绍了Linux下进程的创建,管理,陆续介绍了进程间通信的方式:管道、内存映射、共享内存等。这篇文章继续介绍Linux的进程间通信方式消息队列。

前面文章介绍了Linux下进程的创建,管理,陆续介绍了进程间通信的方式:管道、内存映射、共享内存等。这篇文章继续介绍Linux的进程间通信方式消息队列

image-20211216090722856

1. 消息队列介绍

消息队列通过名字字面意思理解就是队列排队-和平常超市买东西排队付款一样结构,消息队列与FIFO很相似,都是一个队列结构,都可以有多个进程往队列里面写信息,多个进程从队列中读取信息。但FIFO需要读、写的两端事先都打开,才能够开始信息传递工作。而消息队列可以事先往队列中写信息,需要时再打开读取信息。

注意事项:

  1. 消息队列属于顺序队列形式的结构,向队列里写的每一条消息,会追加到队列后面,读取一个就从队列里消除一个。
  2. 写函数不会阻塞,除非队列里存放的消息数量已经满了,才会导致写阻塞。
  3. 读函数,从队列里读取不到数据时,会阻塞。

查看当前系统所有的消息队列:

[root@wbyq 20181005]# ipcs -q

------ Message Queues --------
键值    消息队列ID                        使用的字节数量  队里现存的消息数量                
key        msqid      owner      perms     used-bytes   messages    
0x000004d3 0          root       666        65532        192         
0x00003044 32769      root       666        65424        564         
0x0a120534 65538      root       0          2352         21          
0x0a00000f 196611     root       0          65520        2730        
0xffffffff 163844       root       0          0            0  

查看消息队列的详细信息:

[root@wbyq 20181005]# ipcs -l

------ Shared Memory Limits --------
max number of segments = 4096
max seg size (kbytes) = 4194303
max total shared memory (kbytes) = 1073741824
min seg size (bytes) = 1

------ Semaphore Limits --------
max number of arrays = 128
max semaphores per array = 250
max semaphores system wide = 32000
max ops per semop call = 32
semaphore max value = 32767

------ Messages: Limits --------
max queues system wide = 1736        【系统最多的消息队列数量(最多支持同时1736个消息队列)】
max size of message (bytes) = 65536     【单个消息的最大字节数】
default max size of queue (bytes) = 65536 【默认的单个队列的大小65536】

默认单个队列总字节大小为: 65535字节。
消息队列的最大字节数量和消息的最大条数加起来的总字节数不能超过65535字节。

System V IPC机制消息队列相关的函数接口: 这里先列出所有函数,下面会详细介绍每个函数的用法。

#include <sys / types.h>
#include <sys / ipc.h>
#include <sys / msg.h>
int msgget(key_t key, int msgflg);
int msgsnd(int msqid, struct msgbuf * msgp, size_t msgsz, int msgflg);
ssize_t msgrcv(int msqid, struct msgbuf * msgp, size_t msgsz, long msgtyp, int msgflg);
int msgctl(int msqid, int cmd, struct msqid_ds * buf);

2. 消息队列相关函数介绍

2.1 msgget函数

函数原型:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgget(key_t key, int msgflg);

功能
msgget用于创建和访问一个消息队列。
参数
(1) key:是唯一标识一个消息队列的关键字,如果为IPC_PRIVATE(值为0,用创建一个只有创建者进程才可以访问的消息队列),表示创建一个只由调用进程使用的消息队列,非0值的key(可以通过ftok函数获得)表示创建一个可以被多个进程共享的消息队列;
(2) msgflg:指明队列的访问权限和创建标志,创建标志的可选值为IPC_CREAT和IPC_EXC,如果单独指定IPC_CREAT,msgget要么返回新创建的消息队列id,要么返回具有相同key值的消息队列id;如果IPC_EXCL和IPC_CREAT同时指明,则要么创建新的消息队列,要么当队列存在时,调用失败并返回-1。

/*1. 创建消息队列*/
int msgid = msgget((key_t)1235,0666 | IPC_CREAT);

返回值
成功执行时,返回消息队列标识值(0也是成功的)。

失败返回-1,errno被设为以下的某个值。

EACCES:指定的消息队列已存在,但调用进程没有权限访问它,而且不拥有CAP_IPC_OWNER权能
EEXIST:key指定的消息队列已存在,而msgflg中同时指定IPC_CREAT和IPC_EXCL标志
ENOENT:key指定的消息队列不存在同时msgflg中不指定IPC_CREAT标志
ENOMEM:需要建立消息队列,但内存不足
ENOSPC:需要建立消息队列,但已达到系统的最大消息队列容量

2.2 msgsnd和msgrcy函数

原型:

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

功能

函数msgsnd和msgrcy用来将消息添加到消息队列中和从一个消息队列中获取信息。

参数

(1)msgid:指明消息队列的ID; 通常是msgget函数成功的返回值。

(2)msgbuf:是消息结构体,它的长度必须小于系统规定的上限,必须以一个长整型成员变量开始,接收函数将用这个成员变量来确定消息的类型。必须重写这个结构体,其中第一个参数类型不能改,其他可以自定义。

如下:

struct msgbuf {
    long mtype;         /* type of message */
    char mtext[1];       /* message text */
};

字段mtype是用户自己指定的消息类型(通常是1—5中的任意一个数值),该结构体第2个成员仅仅是一种说明性的结构,实际上用户可以使用任何类型的数据,就是消息内容;

(3)msgsz是消息体的大小,每个消息体最大不要超过4K; 不含消息类型占用的4个字节,即mtext的长度

(4)msgtyp有3种选项:


msgtyp == 0      接收队列中的第1个消息,不区分消息类型
msgtyp > 0        接收对列中的第1个类型等于msgtyp的消息
msgtyp < 0        接收其类型小于或等于msgtyp绝对值的第1个最低类型消息

(5)msgflg有3种选项:

0:当消息队列满时,msgsnd将会阻塞,直到消息能写进消息队列
IPC_NOWAIT:当消息队列已满的时候,msgsnd函数不等待立即返回
IPC_NOERROR:若发送的消息大于size字节,则把该消息截断,截断部分将被丢弃,且不通知发送进程。

2.3 msgctl函数

原型:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgctl(int msqid, int cmd, struct msqid_ds *buf);

功能

msgctl是消息队列的控制函数,常用来删除消息队列。

参数

(1)msqid:由msgget返回的消息队列标识符。

(2)cmd:通常为IPC_RMID表示删除消息队列。

(3)参数buf通常为NULL。

通过命令查看系统消息信息

(1)ipcs -q 命令查看系统的消息队列

(2)ipcs -m查看系统的共享内存

(3)ipcs -s 查看系统的信号量集。

image-20211216085707736

3. 案例代码: 消息队列示例1

下面两个程序分别编译,依次运行,不分先后顺序,就可以看到效果了。

3.1 发送消息

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/stat.h>
#define BUFFER 255
struct msgtype {    //重新定义该结构体
    long mtype;     //第一个参数类型不变
    char buf[BUFFER];
};

int main(int argc,char **argv)
{
    if(argc!=3)
    {
        printf("./app <消息> <消息类型-整数>\n");
        return 0;
    }
    /*1. 创建消息队列*/
    int msgid = msgget((key_t)1235,0666 | IPC_CREAT); 
    /*2. 向消息队列发送消息*/
    struct msgtype msg;
    memset(&msg,0,sizeof(struct msgtype));
    msg.mtype = atoi(argv[2]);                      //给结构体的成员赋值
    strncpy(msg.buf,argv[1],BUFFER);
    msgsnd(msgid,&msg,sizeof(struct msgtype),0);  
    return 0;
}

3.2 读取消息

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/stat.h>
#include <sys/msg.h>
#define BUFFER 255
struct msgtype {
    long mtype;
    char buf[BUFFER];
};

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("./app <消息类型-整数>\n");
        return 0;
    }
    /*1. 创建消息队列-如果存在就打开消息队列*/
    int msgid = msgget((key_t)1235, 0666 | IPC_CREAT); //获得消息队列
    struct msgtype msg;
    memset(&msg,0,sizeof(struct msgtype));
    while(1)
    {
        /*2. 从消息队列里读取指定消息类型*/
        msgrcv(msgid,&msg,sizeof(struct msgtype),atoi(argv[1]),0);
        printf("读取的消息: %s\n", msg.buf);
    }
    return 0;
}

4. 案例代码: 消息队列基本使用

下面两个例子,一个例子用于创建队列,并向队列里写数据,另一个例子从队列里读取数据。

4.1 向队列写入消息

程序运行需要传入两个额外的参数。一个是消息的类型,一个是具体的内容。

./app 1 hello
./app 2 123456789

示例代码:

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <dirent.h>
#include <stdlib.h>
#include <sys/mman.h>
#include <sys/wait.h>
#include <sys/shm.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>

struct msgbuf
{
    long mtype;       /* message type, must be > 0 */
    char mtext[1024];    /* message data */
};

int main(int argc,char **argv)
{
    if(argc!=3)
    {
        printf("./app <消息类型> <消息内容>\n");
        return 0;
    }
    /*1. 创建消息队列*/
    int msqid=msgget(123456,0666|IPC_CREAT);
    /*2. 向消息队列里加入数据*/
    struct msgbuf msg_buf;
    msg_buf.mtype=atoi(argv[1]); //消息类型  1、2、3、4、5
    strcpy(msg_buf.mtext,argv[2]); //消息内容
    msgsnd(msqid,&msg_buf,sizeof(struct msgbuf),0);
    printf("消息发送成功:%s,%s\n",argv[1],argv[2]);
    return 0;
}

4.2 从队列读取消息

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <dirent.h>
#include <stdlib.h>
#include <sys/mman.h>
#include <sys/wait.h>
#include <sys/shm.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>

struct msgbuf
{
    long mtype;       /* message type, must be > 0 */
    char mtext[1024];    /* message data */
};

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("./app <消息类型>\n");
        return 0;
    }
    /*1. 创建消息队列*/
    int msqid=msgget(123456,0666|IPC_CREAT);
    /*2. 从消息队列里取出数据*/
    struct msgbuf msg_buf;
    msgrcv(msqid,&msg_buf,sizeof(struct msgbuf),atoi(argv[1]),0);
    printf("消息读取成功:%d,%s\n",msg_buf.mtype,msg_buf.mtext);
    return 0;
}
目录
相关文章
|
21天前
|
存储 缓存 监控
Linux缓存管理:如何安全地清理系统缓存
在Linux系统中,内存管理至关重要。本文详细介绍了如何安全地清理系统缓存,特别是通过使用`/proc/sys/vm/drop_caches`接口。内容包括清理缓存的原因、步骤、注意事项和最佳实践,帮助你在必要时优化系统性能。
158 78
|
1天前
|
消息中间件 Linux
Linux:进程间通信(共享内存详细讲解以及小项目使用和相关指令、消息队列、信号量)
通过上述讲解和代码示例,您可以理解和实现Linux系统中的进程间通信机制,包括共享内存、消息队列和信号量。这些机制在实际开发中非常重要,能够提高系统的并发处理能力和数据通信效率。希望本文能为您的学习和开发提供实用的指导和帮助。
35 20
|
25天前
|
Linux Shell 网络安全
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
本指南介绍如何利用 HTA 文件和 Metasploit 框架进行渗透测试。通过创建反向 shell、生成 HTA 文件、设置 HTTP 服务器和发送文件,最终实现对目标系统的控制。适用于教育目的,需合法授权。
56 9
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
|
21天前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
89 13
|
22天前
|
Ubuntu Linux C++
Win10系统上直接使用linux子系统教程(仅需五步!超简单,快速上手)
本文介绍了如何在Windows 10上安装并使用Linux子系统。首先,通过应用商店安装Windows Terminal和Linux系统(如Ubuntu)。接着,在控制面板中启用“适用于Linux的Windows子系统”并重启电脑。最后,在Windows Terminal中选择安装的Linux系统即可开始使用。文中还提供了注意事项和进一步配置的链接。
40 0
|
7月前
|
监控 Linux 应用服务中间件
探索Linux中的`ps`命令:进程监控与分析的利器
探索Linux中的`ps`命令:进程监控与分析的利器
146 13
|
6月前
|
运维 关系型数据库 MySQL
掌握taskset:优化你的Linux进程,提升系统性能
在多核处理器成为现代计算标准的今天,运维人员和性能调优人员面临着如何有效利用这些处理能力的挑战。优化进程运行的位置不仅可以提高性能,还能更好地管理和分配系统资源。 其中,taskset命令是一个强大的工具,它允许管理员将进程绑定到特定的CPU核心,减少上下文切换的开销,从而提升整体效率。
掌握taskset:优化你的Linux进程,提升系统性能
|
6月前
|
弹性计算 Linux 区块链
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
203 4
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
|
5月前
|
算法 Linux 调度
探索进程调度:Linux内核中的完全公平调度器
【8月更文挑战第2天】在操作系统的心脏——内核中,进程调度算法扮演着至关重要的角色。本文将深入探讨Linux内核中的完全公平调度器(Completely Fair Scheduler, CFS),一个旨在提供公平时间分配给所有进程的调度器。我们将通过代码示例,理解CFS如何管理运行队列、选择下一个运行进程以及如何对实时负载进行响应。文章将揭示CFS的设计哲学,并展示其如何在现代多任务计算环境中实现高效的资源分配。
|
6月前
|
存储 缓存 安全
【Linux】冯诺依曼体系结构与操作系统及其进程
【Linux】冯诺依曼体系结构与操作系统及其进程
186 1