APUE_消息队列 女爷i 2022-08-28 06:58 201阅读 0赞 ### 1.引言 ### 消息队列是消息的链接表,存放在内核中并由消息队列标识符标识。 ### 2.创建消息队列 ### 每一个队列都有一个msqid\_ds结构与其相关联。此结构规定了消息队列的当前状态。主要包括XSI IPC共同的信息ipc\_perm,队列长度,队列中消息数,和最后对队列进行操作的进程及时间。 struct msqid_ds { struct ipc_perm msg_perm; msgqnum_t msg_qnum; /* # of messages on queue */ msglen_t msg_qbytes; /* max # of bytes on queue */ pid_t msg_lspid; /* pid of last msgsnd() */ pid_t msg_lrpid; /* pid of last msgrcv() */ time_t msg_stime; /* last-msgsnd() time */ time_t msg_rtime; /* last-msgrcv() time */ time_t msg_ctime; /* last-change time */ . . . }; 使用消息队列第一个要用到的函数就是msgget,msget函数用于创建一个新队列或打开一个现存的队列。 #include <sys/msg.h> int msgget(key_t key, int flag); //返回值:成功则返回消息队列ID,出错返回-1 对消息队列的操作需要使用,消息队列ID。 ### 3.操作消息队列 ### msgctl函数对消息队列执行多种操作。 #include <sys/msg.h> int msgctl(int msqid, int cmd, struct msqid_ds *buf ); //成功为0,出错为-1 cmd参数说明对由msgid指定的消息队列要执行的命令。 IPC\_STAT,取此消息队列的msqid\_ds结构,并将其放到buf指向的结构中。 IPC\_SET,按由buf指向结构中的值,设置与此队列相关结构中的四个字段:msg\_perm.uid、msg\_perm.gid、msg\_perm.mode和msg\_qbytes。调用进程的有效用户ID必须等于msg\_perm.cuid或msd\_perm.uid,或者是具有超级用户权限的进程。只有超级用户才能增加msg\_qbytes的值。 **IPC\_RMID****,从系统中删除该消息队列以及仍在队列中的所有数据。(XSI IPC都是没有引用计数的!!!)**这种删除立即生效。调用进程的有效用户ID必须等于msg\_perm.cuid或msd\_perm.uid,或者是具有超级用户权限的进程。 这三条命令(IPC\_STAT、IPC\_SET、IPC\_RMID)也可用于信号量或者共享存储。 ### 4.发送消息 ### msgsnd函数可将数据发送到消息队列尾端。 #include <sys/msg.h> int msgsnd(int msqid, const void *ptr, size_t nbytes, int flag); //成功为0,出错为-1 每一个消息由三部分组成:正长整型的类型字段mtype、非负长度nbytes以及实际数据字节(对应于长度)。参数ptr是一个结构的指针,它包含了消息类型mtype和消息数据。这样接受者可以使用消息类型以非先进先出的次序取消息。 struct mymesg { long mtype; /* positive message type */ char mtext[nbytes]; /* message data, of length nbytes */ }; 当msgsnd成功返回,与消息队列关联的msqid\_ds结构得到更新,以标明发出该调用的进程ID(msg\_lspid)、进行该调用的时间(msg\_stime),并指示队列中增加了一条消息(msg\_qnum)。 ### 5.接收消息 ### msgrcv函数可从消息队列中取得消息。 #include <sys/msg.h> ssize_t msgrcv(int msqid, void *ptr, size_t nbytes, long type, int flag); //若成功则返回消息的数据部分的长度,出错返回-1 参数ptr与msgsnd中的一样。nbytes说明数据缓冲区的长度。若返回的消息大于nbytes,而且在flag中设置了MSG\_NOERROR,则该消息被截断。在这种情况下,消息的截断部分被丢弃。如果没有设置这一标志,而消息又太长,则出错返回E2BIG,消息仍然留在队列中。 参数type指定想要接收哪一种消息:1)type==0,返回队列中的第一个消息;2)type>0,返回队列中消息类型为type的第一个消息;3)type<0,返回队列中消息类型值小于或等于type绝对值的消息,如果这种消息有若干个,则取类型值最小的消息。**这就可以实现以非先进先出的方式操作消息队列!!!** msgrcv成功执行时,内核更新与消息队列关联的msqid\_ds结构,以指示调用者的ID(msg\_lrpid)和调用时间(msg\_rtime),并将队列中的消息数(msg\_qnum)减1。 ### 6.消息队列的劣势 ### 消息队列原来的实施目的就是提供比一般IPC更高速度的进程通信方式,不过现在与其他IPC相比在速度上没有什么区别了,甚至STREAM管道要比消息队列更加快。而且消息队列具有XSI IPC共有的问题,**所以一般不采用消息队列作为进程间通信的手段。** ### 7.示例 ### #include <stdio.h> #include <stdlib.h> #include <sys/ipc.h> #include <sys/msg.h> #include <time.h> #include "apue.h" struct mymesg { long mtype; char mtext[6]; }; void print_msqid(const struct msqid_ds *ds) { printf("uid=%d, gid=%d, cuid=%d, cgid=%d, mode=%o/n", ds->msg_perm.uid, ds->msg_perm.gid, ds->msg_perm.cuid, ds->msg_perm.cgid, ds->msg_perm.mode); printf("msg_qnum=%d, msg_qbytes=%d/n", (int)ds->msg_qnum, (int)ds->msg_qbytes); printf("msg_lspid=%d, msg_lrpid=%d/n", ds->msg_lspid, ds->msg_lrpid); printf("msg_stime=%smsg_rtime=%smsg_ctime=%s", ctime(&ds->msg_stime), ctime(&ds->msg_rtime), ctime(&ds->msg_ctime)); } int main() { pid_t pid; key_t key; int msgid; struct mymesg snd_msg={1, "hello"}; struct mymesg rcv_msg; struct msqid_ds msqds; if((key = ftok("temp", 'b')) == -1) err_sys("ftok error"); if((pid = fork()) < 0) err_sys("fork error"); else if(pid == 0) /*parent*/ { if((msgid = msgget(key, IPC_CREAT | 00666)) == -1) err_sys("create msgqueue error"); if(msgsnd(msgid, &snd_msg, 6, 0) == -1) err_sys("msgsnd error"); exit(0); } if(waitpid(pid, NULL, 0) == -1) err_sys("waitpid error"); if((msgid = msgget(key, 0)) == -1) err_sys("open msgqueue error"); if(msgrcv(msgid, &rcv_msg, 6, 1, 0) == -1) err_sys("msgrcv error"); fputs(rcv_msg.mtext, stdout); fputc('/n', stdout); if(msgctl(msgid, IPC_STAT, &msqds) == -1) err_sys("msgctl IPC_STAT error"); print_msqid(&msqds); if(msgctl(msgid, IPC_RMID, NULL) == -1) err_sys("msgctl IPC_RMID error"); exit(0); }
相关 消息队列 一、消息队列MQ(Message Queue): 1)消息队列是一种先进先出的数据结构; 2)消息队列使用的“协议”不是具体的通讯协议,而是更高层次通讯模型。它定义 「爱情、让人受尽委屈。」/ 2023年10月11日 11:11/ 0 赞/ 74 阅读
相关 消息队列? 对于 MQ 来说,其实不管是 RocketMQ、Kafka 还是其他消息队列,它们的本质都是:一发一存一消费。下面我们以这个本质作为根,一起由浅入深地聊聊 MQ。 01 从 Bertha 。/ 2023年10月10日 12:49/ 0 赞/ 52 阅读
相关 消息队列 一、什么是消息队列 以下为虚构的小故事: 有一天,产品跑来跟小王说:“我们要做一个用户实名的功能,需要在用户实名成功后给用户发一条短信。” 小王(攻城狮leve 缺乏、安全感/ 2023年10月06日 17:00/ 0 赞/ 70 阅读
相关 消息队列 1. 消息队列在项目中的使用 背景:在分布式系统中是如何处理高并发的。 由于在高并发的环境下,来不及同步处理用户发送的请求,则会导致请求发生阻塞。比如说,大量的ins ﹏ヽ暗。殇╰゛Y/ 2022年12月15日 03:23/ 0 赞/ 292 阅读
相关 APUE_消息队列 1.引言 消息队列是消息的链接表,存放在内核中并由消息队列标识符标识。 2.创建消息队列 每一个队列都有一个msqid\_d 女爷i/ 2022年08月28日 06:58/ 0 赞/ 202 阅读
相关 消息队列 https://www.cnblogs.com/457248499-qq-com/p/7392678.html 来源 消息队列:在消息的传输过程中保存消息的容器。 迈不过友情╰/ 2022年05月30日 07:42/ 0 赞/ 306 阅读
相关 消息队列 消息队列是啥?我觉得大家都心知肚明,已经众所周知到不用解释的程度。不过,但凡学习、解释一样东西,都应该遵循 “它是什么?”、 “做什么用?”、 “为啥要用它” ╰半橙微兮°/ 2022年04月23日 08:42/ 0 赞/ 380 阅读
相关 消息队列 消息队列介绍 维基百科上的描述:在计算机科学中,消息队列(Message queue)是一种进程间通信或同一进程的不同线程间的通信方式,软件的贮列用来处理一系列的输入,通 红太狼/ 2022年01月26日 10:51/ 0 赞/ 366 阅读
相关 消息队列 为什么写这篇文章? 博主有两位朋友分别是小A和小B: 1. 小A,工作于传统软件行业(某社保局的软件外包公司),每天工作内容就是和产品聊聊需求,改改业务逻辑。再不然就 小咪咪/ 2021年12月20日 06:51/ 0 赞/ 460 阅读
还没有评论,来说两句吧...