IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 系统运维 -> Linux进程间通信(四)使用消息队列 -> 正文阅读

[系统运维]Linux进程间通信(四)使用消息队列

一、概述

????????消息队列提供了一种从一个进程向另一个进程发送一个数据块的方法。 ?每个数据块都被认为含有一个类型,接收进程可以独立地接收含有不同类型的数据结构。我们可以通过发送消息来避免命名管道的同步和阻塞问题。但是消息队列与命名管道一样,每个数据块都有一个最大长度的限制。

二、函数讲解

1、ftok函数

????????该函数将文件路径和文件描述符转换为一个键值key(也可以自己定义一个键值,只要和其他不冲突就好)

key_t ftok(const char *fname, int id);

?????????fname:必须是一个已经存在且程序可范围的文件。

?????????idid是子序号。虽然是int类型,但是只使用8bits(1-255)。

????????返回值:返回一个key(键值)。

2、msgget函数

????????该函数用来创建和访问一个消息队列,它的原型为:

int?msgget(key_t, key, int msgflg);

????????key:函数ftok的返回值或IPC_PRIVATE。

????????msgflag: IPC_CREAT:创建新的消息队列。

????????返回值:调用成功返回队列标识符,否则返回-1.

3、msgsnd函数:

????????该函数用于发送消息到消息队列中,它的原型为:

int?msgsend(int msgid, const void *msg_ptr, size_t msg_sz, int msgflg);

????????msqid:消息队列的识别码

????????msg_ptr:指向消息缓冲区的指针,此位置用来暂时存储发送和接收的消息,是一个用户可定义的通用结构,形态如下

struct msgbuf {
    long msgtype; /* 消息类型,必须 > 0 */
    char buf[BUF_SIZE]; /* 消息文本 */
};

????????msg_sz:消息的大小。

????????msgflg:这个参数依然是控制函数行为的标志,取值可以是:0,表示忽略;IPC_NOWAIT,如果消息队列为空,则返回一个ENOMSG,并将控制权交回调用函数的进程。如果不指定这个参数,那么进程将被阻塞直到函数可以从队列中得到符合条件的消息为止。如果一个client 正在等待消息的时候队列被删除,EIDRM 就会被返回。如果进程在阻塞等待过程中收到了系统的中断信号,EINTR 就会被返回。MSG_NOERROR,如果函数取得的消息长度大于msgsz,将只返回msgsz 长度的信息,剩下的部分被丢弃了。如果不指定这个参数,E2BIG 将被返回,而消息则留在队列中不被取出。当消息从队列内取出后,相应的消息就从队列中删除了。

4、msgrcv函数

????????该函数用于从消息队列中获取消息,它的原型为:

int?msgrcv(int msgid, void *msg_ptr, size_t msg_sz, long int msgtype, int msgflg);

????????msqid:消息队列的识别码

????????msg_ptr:指向消息缓冲区的指针,此位置用来暂时存储发送和接收的消息,是一个用户可定义的通用结构,形态如下:

struct msgbuf {
    long msgtype; /* 消息类型,必须 > 0 */
    char buf[BUF_SIZE]; /* 消息文本 */
};

????????msg_sz:消息的大小。

????????msgtype:消息类型,msgtype等于0 则返回队列的最早的一个消息;msgtype大于0,则返回其类型为msgtype的第一个消息;msgtype小于0,则返回其类型小于或等于msgtype参数的绝对值的最小的一个消息。

????????msgflg:这个参数依然是控制函数行为的标志,取值可以是:0,表示忽略;IPC_NOWAIT,如果消息队列为空,则返回一个ENOMSG,并将控制权交回调用函数的进程。如果不指定这个参数,那么进程将被阻塞直到函数可以从队列中得到符合条件的消息为止。如果一个client 正在等待消息的时候队列被删除,EIDRM 就会被返回。如果进程在阻塞等待过程中收到了系统的中断信号,EINTR 就会被返回。MSG_NOERROR,如果函数取得的消息长度大于msgsz,将只返回msgsz 长度的信息,剩下的部分被丢弃了。如果不指定这个参数,E2BIG 将被返回,而消息则留在队列中不被取出。当消息从队列内取出后,相应的消息就从队列中删除了。

5、msgctl函数

????????该函数用来控制消息队列,它与共享内存的shmctl函数相似,它的原型为:

int?msgctl(int?msgid,?int?cmd,?struct?msgid_ds *buf);

????????cmd:是将要采取的动作,它可以取3个值:

????????IPC_STAT:把msgid_ds结构中的数据设置为消息队列的当前关联值,即用消息队列的当前关联值覆盖msgid_ds的值。

????????IPC_SET:如果进程有足够的权限,就把消息列队的当前关联值设置为msgid_ds结构中给出的值。

????????IPC_RMID:删除消息队列。

????????buf:指向msgid_ds结构的指针,它指向消息队列模式和访问权限的结构。msgid_ds结构至少包括以下成员:

struct msgid_ds
{
    uid_t shm_perm.uid;
    uid_t shm_perm.gid;
    mode_t shm_perm.mode;
};

????????返回值:成功时返回0,失败时返回-1。

三、使用消息队列进行进程间通信

?????????1、接收信息的程序源文件为msgrecv.c的源代码为:

/*使用消息队列通信*/	
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <sys/msg.h>

#define BUF_SIZE 	1024
struct msg_st
{
	long int msg_type;
	char buf[BUF_SIZE];
};
int main()
{
	int is_run = 1;
	int msgid = -1;
	struct msg_st data;
	long int msgtype = 0;
	
	//建立消息队列
	msgid = msgget((key_t)1234, 0666 | IPC_CREAT);
	if(msgid == -1)
	{
		fprintf(stderr, "msgget failed with error: %d\n", errno);
		exit(EXIT_FAILURE);
	} 
	//从队列中获取消息,直到遇到end消息为止
	while(is_run)
	{
		if(msgrcv(msgid, (void*)&data, BUF_SIZE, msgtype, 0) == -1)
		{
			fprintf(stderr, "msgrcv failed with errno: %d\n", errno);
			exit(EXIT_FAILURE);
		}
		printf("Receive data buf is: %s\n",data.buf);
		
		//遇到end结束
		if(strncmp(data.buf,"end", 3) == 0)
		is_run = 0;
	}
	
	//删除消息队列
	if(msgctl(msgid, IPC_RMID, 0) == -1)
	{
		fprintf(stderr, "msgctl(IPC_RMID) failed\n");
		exit(EXIT_FAILURE);
	}
	exit(EXIT_SUCCESS);
}

????????2、发送信息的程序的源文件msgsend.c的源代码为:

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/msg.h>
#include <errno.h>
 
#define BUF_SIZE 1024
struct msg_st
{
	long int msg_type;
	char buf[BUF_SIZE];
};
 
int main()
{
	int is_run = 1;
	struct msg_st data;
	char buffer[BUFSIZ];
	int msgid = -1;
 
	//建立消息队列
	msgid = msgget((key_t)1234, 0666 | IPC_CREAT);
	if(msgid == -1)
	{
		fprintf(stderr, "msgget failed with error: %d\n", errno);
		exit(EXIT_FAILURE);
	}
	
	//向消息队列中写消息,直到写入end
	while(is_run)
	{
		//输入数据
		printf("input data buf is: ");
		fgets(buffer, BUFSIZ, stdin);
		data.msg_type = 1;    //注意2
		strcpy(data.buf, buffer);
		
		//向队列发送数据
		if(msgsnd(msgid, (void*)&data, BUF_SIZE, 0) == -1)
		{
			fprintf(stderr, "msgsnd failed\n");
			exit(EXIT_FAILURE);
		}
		
		//输入end结束输入
		if(strncmp(buffer, "end", 3) == 0)
		{
			is_run = 0;
		}
		sleep(1);
	}
	exit(EXIT_SUCCESS);
}

四、编译结果

????????编译结果如下图

  系统运维 最新文章
配置小型公司网络WLAN基本业务(AC通过三层
如何在交付运维过程中建立风险底线意识,提
快速传输大文件,怎么通过网络传大文件给对
从游戏服务端角度分析移动同步(状态同步)
MySQL使用MyCat实现分库分表
如何用DWDM射频光纤技术实现200公里外的站点
国内顺畅下载k8s.gcr.io的镜像
自动化测试appium
ctfshow ssrf
Linux操作系统学习之实用指令(Centos7/8均
上一篇文章      下一篇文章      查看所有文章
加:2021-12-26 22:38:51  更:2021-12-26 22:39:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 10:56:25-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码