百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Linux中的消息队列、共享内存,你确定都掌握了吗?

off999 2025-01-24 13:25 68 浏览 0 评论

消息队列(message queue)

消息队列是消息的链表,存放在内存中,由内核维护

消息队列的特点
1、消息队列中的消息是有类型的。
2、消息队列中的消息是有格式的。
3、消息队列可以实现消息的随机查询。消息不一定要以先进先出的次序读取,编程时可以按消息的类型读取。
4、消息队列允许一个或多个进程向它写入或者读取消息。
5、与无名管道、命名管道一样,从消息队列中读出消息,消息队列中对应的数据都会被删除。
6、每个消息队列都有消息队列标识符,消息队列的标识符在整个系统中是唯一的。
7、只有内核重启或人工删除消息队列时,该消息队列才会被删除。若不人工删除消息队列,消息队列会一直存在于系统中。
8、*消息队列可以独立于进程存在,可以无阻塞收发,可以选择性的接收消息
》 在ubuntu 12.04中消息队列限制值如下:

  • 每个消息内容最多为8K字节
  • 每个消息队列容量最多为16K字节
  • 系统中消息队列个数最多为1609个
  • 系统中消息个数最多为16384个

System V提供的IPC通信机制需要一个key值,通过key值就可在系统内获得一个唯一的消息队列标识符。
key值可以是人为指定的,也可以通过ftok函数获得。

#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok(const char *pathname, int proj_id);
功能:
	获得项目相关的唯一的IPC键值。
参数:
	pathname:路径名
	proj_id:项目ID,非0整数(只有低8位有效)
返回值:
	成功返回key值
	失败返回 -1

创建消息队列:

#include <sys/msg.h>
int msgget(key_t key, int msgflg);
功能:
创建一个新的或打开一个已经存在的消息队列。不同的进程调用此函数,
只要用相同的key值就能得到同一个消息队列的标识符。
参数:
key:IPC键值。
msgflg:标识函数的行为及消息队列的权限。
返回值:
成功:消息队列的标识符,失败:返回-1。

》msgflg的取值:
IPC_CREAT:创建消息队列。
IPC_EXCL:检测消息队列是否存在。
位或权限位:消息队列位或权限位后可以设置消息队列的访问权限,
格式和open函数的mode_t一样,但可执行权限未使用。

使用shell命令操作消息队列:
查看消息队列
ipcs -q
删除消息队列
ipcrm -q msqid

消息队列的消息的格式:

typedef struct _msg
{
	long mtype; /*消息类型*/
	char mtext[100]; /*消息正文*/
	... /*消息的正文可以有多个成员*/
}MSG;
消息类型必须是长整型的,而且必须是结构体类型的第一个成员,
类型下面是消息正文,正文可以有多个成员(正文成员可以是任意数据类型的)。

发送消息:

#include <sys/msg.h>
int msgsnd(int msqid, const void *msgp,size_t msgsz, int msgflg);
功能:
	将新消息添加到消息队列。
参数:
	msqid:消息队列的标识符。
	msgp:待发送消息结构体的地址。
	msgsz:消息正文的字节数。
	msgflg:函数的控制属性
		0:msgsnd调用阻塞直到条件满足为止。
		IPC_NOWAIT: 若消息没有立即发送则调用该函数的进程会立即返回。
返回值:
	成功:0
	失败:返回-1

接收消息:

#include <sys/msg.h>
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz,
long msgtyp, int msgflg);
功能:
从标识符为msqid的消息队列中接收一个消息。一旦接收消息成功,
则消息在消息队列中被删除。
参数:
msqid:消息队列的标识符,代表要从哪个消息列中获取消息。
msgp: 存放消息结构体的地址。
msgsz:消息正文的字节数。
msgtyp:消息的类型、可以有以下几种类型
	msgtyp = 0:返回队列中的第一个消息
	msgtyp > 0:返回队列中消息类型为msgtyp的消息
	msgtyp < 0:返回队列中消息类型值小于或等于
	msgtyp绝对值的消息,如果这种消息有若干个,则取类型值最小的消息。
	注意:
	若消息队列中有多种类型的消息,msgrcv获取消息的时候按消息类型获取,
	不是先进先出的。在获取某类型消息的时候,若队列中有多条此类型的消息,
	则获取最先添加的消息,即先进先出原则。
msgflg:函数的控制属性
	0:msgrcv调用阻塞直到接收消息成功为止。
	MSG_NOERROR:若返回的消息字节数比nbytes字节数多,则消息就会截短到nbytes字节,且不通知消息发送进程。
	IPC_NOWAIT:调用进程会立即返回。若没有收到消息则立即返回-1。
返回值:
	成功返回读取消息的长度
	失败返回-1。

消息队列的控制:

#include <sys/msg.h>
int msgctl(int msqid, int cmd, struct msqid_ds *buf);
功能:
	对消息队列进行各种控制,如修改消息队列的属性,或删除消息消息队列。
参数:
	msqid:消息队列的标识符。
	cmd:函数功能的控制。
	buf:msqid_ds数据类型的地址,用来存放或更改消息队列的属性。
返回值:
	成功:返回 0
	失败:返回 -1

》cmd:函数功能的控制
	IPC_RMID:删除由msqid指示的消息队列,将它从系统中删除并破坏相关数据结构。
	IPC_STAT:将msqid相关的数据结构中各个元素的当前值存入到由buf指向的结构中。
	IPC_SET:将msqid相关的数据结构中的元素设置为由buf指向的结构中的对应值。

文章福利:现在C++程序员面临的竞争压力越来越大。那么,作为一名C++程序员,怎样努力才能快速成长为一名高级的程序员或者架构师,或者说一名优秀的高级工程师或架构师应该有怎样的技术知识体系,这不仅是一个刚刚踏入职场的初级程序员,也是工作三五年之后开始迷茫的老程序员,都必须要面对和想明白的问题。为了帮助大家少走弯路,技术要做到知其然还要知其所以然。以下视频获取点击:正在跳转

如果想学习C++工程化、高性能及分布式、深入浅出。性能调优、TCP,协程,Nginx源码分析Nginx,ZeroMQ,MySQL,Redis,MongoDB,ZK,Linux内核,P2P,K8S,Docker,TCP/IP,协程,DPDK的朋友可以看一下这个学习地址C/C++Linux服务器开发/后台架构师【零声教育】-学习视频教程-腾讯课堂

例:01_message_queue_write.c 01_message_queue_read.c

总结:

》每个程序都有两个任务,一个任务是负责接收消息,一个任务是负责发送消息,
通过fork创建-子进程实现多任务。
》一个进程负责接收信息,它只接收某种类型的消息,只要别的进程发送此类型的消息,
此进程-就能收到。收到后通过消息的name成员就可知道是谁发送的消息。
》另一个进程负责发信息,可以通过输入来决定发送消息的类型。
》设计程序的时候,接收消息的进程接收消息的类型不一样,这样就实现了发送的消息
只被接收-此类型消息的人收到,其它人收不到。这样就是实现了给特定的人发送消息。

共享内存(shared memory)

共享内存允许两个或者多个进程共享给定的存储区域。

共享内存的特点
1、共享内存是进程间共享数据的一种最快的方法。一个进程向共享的内存区域写入了数据,共享这个内存区域的所有进程就可以立刻看到其中的内容。
2、使用共享内存要注意的是多个进程之间对一个给定存储区访问的互斥。若一个进程正在向共享内存区写数据,则在它做完这一步操作前,别的进程不应当去读、写这些数据。


在ubuntu 12.04中共享内存限制值如下:
-共享存储区的最小字节数:1
-共享存储区的最大字节数:32M
-共享存储区的最大个数:4096
-每个进程最多能映射的共享存储区的个数:4096

获得一个共享存储标识符

#include <sys/ipc.h>
#include <sys/shm.h>
int shmget(key_t key, size_t size,int shmflg);
功能:创建或打开一块共享内存区
参数:
key:IPC键值
size:该共享存储段的长度(字节)
shmflg:标识函数的行为及共享内存的权限。
返回值:
成功:返回共享内存标识符。
失败:返回-1。

》参数属性shmflg:
IPC_CREAT:如果不存在就创建
IPC_EXCL:如果已经存在则返回失败
位或权限位:共享内存位或权限位后可以设置共享内存的访问权限,格式和open函数的mode_t一样,但可执行权限未使用。

使用shell命令操作共享内存:
查看共享内存
ipcs -m
删除共享内存
ipcrm -m shmid

共享内存映射(attach)

#include <sys/types.h>
#include <sys/shm.h>
void *shmat(int shmid, const void *shmaddr,int shmflg);
功能:
将一个共享内存段映射到调用进程的数据段中。
参数:
shmid:共享内存标识符。
shmaddr:共享内存映射地址(若为NULL则由系统自动指定),推荐使用NULL。
shmflg:共享内存段的访问权限和映射条件。
返回值:
成功:返回共享内存段映射地址
失败:返回 -1

》shmflg:共享内存段的访问权限和映射条件
	0:共享内存具有可读可写权限。
	SHM_RDONLY:只读。
	SHM_RND:(shmaddr非空时才有效)
	没有指定SHM_RND则此段连接到shmaddr所指定的地址上(shmaddr必需页对齐)。
	指定了SHM_RND则此段连接到shmaddr-shmaddr%SHMLBA 所表示的地址上。
》注:
shmat函数使用的时候第二个和第三个参数一般设为NULL和0,
即系统自动指定共享内存地址,并且共享内存可读可写。

解除共享内存映射(detach)

#include <sys/types.h>
#include <sys/shm.h>
int shmdt(const void *shmaddr);
功能:
将共享内存和当前进程分离(仅仅是断开联系并不删除共享内存)。
参数:
shmaddr:共享内存映射地址。
返回值:
成功返回 0,
失败返回 -1。

共享内存控制

#include <sys/ipc.h>
#include <sys/shm.h>
int shmctl(int shmid, int cmd,struct shmid_ds *buf);
功能:共享内存空间的控制。
参数:
shmid:共享内存标识符。
cmd:函数功能的控制。
buf:shmid_ds数据类型的地址,用来存放或修改共享内存的属性。
返回值:
成功返回 0,
失败返回 -1。

》cmd:函数功能的控制
	IPC_RMID:删除。
	IPC_SET:设置shmid_ds参数。
	IPC_STAT:保存shmid_ds参数。
	SHM_LOCK:锁定共享内存段(超级用户)。
	SHM_UNLOCK:解锁共享内存段。

例:02_shared_memory_write.c 02_shared_memory_read.c

注意:
SHM_LOCK用于锁定内存,禁止内存交换。并不代表共享内存被锁定后禁止其它进程访问。其真正的意义是:被锁定的内存不允许被交换到虚拟内存中。这样做的优势在于让共享内存一直处于内存中,从而提高程序性能。

相关推荐

win7系统卡顿怎么优化(win7很慢很卡怎么优化)

1、首先打开安全卫士,进入安全卫士首页,单击软件窗口右下角的“更多”图标,打开扩展应用程序。2、单击选择“我的工具”。3、在我的工具菜单里面找到“人工服务”单击打开人工服务。4、在人工服务对话框有很多...

如何查看c盘微信聊天记录(如何查看c盘微信聊天记录内存大小)

微信群中的消息只要没删除基本都能保存,想要找微信群中几个多月前的消息可以直接根据日期来查找聊天记录。操作如下:1、打开想要查找记录的微信群,点击右上角人形图标;2、点击查找聊天内容;3、选择按日...

office2016家庭版激活密钥(office家庭版激活码2019)

走淘宝吧,因为零售版的密钥只能用一次。大概几块钱就能激活2016。如果你不在乎钱的话可以向我一样,订阅一个office365.实在不行可以和几个人一起买一个家庭版的365.出现这个情况,找微软申诉是没...

移动硬盘驱动器下载安装(移动硬盘驱动器下载安装教程)

1、右键单击您的桌面,选择“新建文件夹”,并命名该文件夹(例如“usb驱动程序”);2、然后到本站下载驱动程序;3、将其解压缩至在您的桌面上刚刚创建的usb驱动程序文件夹;4、单击开始菜单,然后选择设...

电脑硬盘格式化工具(电脑 格式化硬盘)

硬盘格式化工具很多,PQMACGIG8.0(中文就叫硬盘分区魔法师)是比较好的一个,这个是在WINDOWS下比叫好用,(个人感觉)FDISK也是比较好的一个,这个一般用在DOS下分区格式化WIN...

photoshop是一款什么软件(ps指的是什么软件)

这个说法是错误的,ps软件“即:photoshop”是由美国著名的“adobe阿多比”公司出品的专业的图像处理软件,它不是由微软公司出品的软件。众所周知的是,微软公司以设计视窗操作系统名满全球,它出...

ipad越狱的好处与坏处(ipad越狱好不好)

  好处一:  1、重命名、重组应用程序  如果你看着Sparrow(iOS最优秀邮件客户端)这个名字不爽,越狱之后就可以改成“Email”,如果你觉得“豆瓣电台”这个名字不给力,那就改成“中央人民广...

win7光盘重装系统步骤图解(win7光盘如何重装系统)

1.确认您的电脑支持从光盘启动。如果支持,可以直接将Windows7安装光盘插入电脑的光驱中。 2.打开电脑,按下F2、F10、F12或Delete等键进入BIOS设置界面。 ...

电脑已联网却无法上网(电脑已经联网了但是不能上网)

电脑连上网后,仍可能存在无法上网的情况,这可能是由多种原因造成的。以下是一些可能的原因和解决方法:1.浏览器问题:有时候,浏览器可能会出现故障,导致无法正常访问网络。您可以尝试清除浏览器的缓存和co...

u盘价格一览表(u盘单价)

不同品牌价格不同,不同内存价格也不同,例如8g、16g、32g、64g等多种容量大小的,根据容量的不同,报价在29元到120元之间不等。闪存盘虽然小,但相对来说却有很大的存储容量。U盘大多能够存储比一...

windows查看ip命令(windows如何查看ip地址)

查看电脑IP:    1)使用Windows+R键打开“运行”窗口,然后输入CMD进入命令提示窗口2)进入命令窗口之后,输入:ipconfig/all回车即可...

内存条的作用(内存条的作用和参数配置)

内存条是存储电脑运行所需的数据和程序,帮助CPU快速读取和运行,提高计算机的运行速度和处理能力。内存条也被称为随机存取存储器(RAM),是电脑中非常必要的一个组件。常见的内存条类型有DDR、DDR2、...

autocad2012安装失败(autocad2012无法安装)

如果您遇到CAD2012安装不了的问题,可能有几个原因导致这种情况。以下是一些常见的解决方法:1.确保系统要求:首先,请确保您的计算机符合CAD2012的系统要求。检查您的操作系统版本、内存、处理器...

win11 16g内存最佳虚拟内存(window10 16个g虚拟内存设置)

内存足够大可以将系统的虚拟内存关掉。1、鼠标右键【此电脑】,在菜单中选择【属性】。2、进入属性后,点击【高级系统设置】。3、进入系统属性后,点击高级下面的【设置】。4、进入性能设置后,点击【高级】。5...

查看windows7激活码(win7激活码哪里看)

windows7激活密钥如下:PPBK3-M92CH-MRR9X-34Y9P-7CH2FQ8JXJ-8HDJR-X4PXM-PW99R-KTJ3H8489X-THF3D-BDJQR-D27PH-P...

取消回复欢迎 发表评论: