《Linux环境进程间通信(三):消息队列.doc》由会员分享,可在线阅读,更多相关《Linux环境进程间通信(三):消息队列.doc(15页珍藏版)》请在三一办公上搜索。
1、本系列文章中的前两部分,我们探讨管道及信号两种通信机制,本文将深入第三部分,介绍系统 V 消息队列及其相应 API。消息队列(也叫做报文队列)能够克服早期unix通信机制的一些缺点。作为早期unix通信机制之一的信号能够传送的信息量有限,后来虽然POSIX 1003.1b在信号的实时性方面作了拓广,使得信号在传递信息量方面有了相当程度的改进,但是信号这种通信方式更像即时的通信方式,它要求接受信号的进程在某个时间范围内对信号做出反应,因此该信号最多在接受信号进程的生命周期内才有意义,信号所传递的信息是接近于随进程持续的概念(process-persistent),见附录 1;管道及有名管道及有名
2、管道则是典型的随进程持续IPC,并且,只能传送无格式的字节流无疑会给应用程序开发带来不便,另外,它的缓冲区大小也受到限制。消息队列就是一个消息的链表。可以把消息看作一个记录,具有特定的格式以及特定的优先级。对消息队列有写权限的进程可以向中按照一定的规则添加新消息;对消息队列有读权限的进程则可以从消息队列中读走消息。消息队列是随内核持续的(参见附录 1)。目前主要有两种类型的消息队列:POSIX消息队列以及系统V消息队列,系统V消息队列目前被大量使用。考虑到程序的可移植性,新开发的应用程序应尽量使用POSIX消息队列。在本系列专题的序(深刻理解Linux进程间通信(IPC)中,提到对于消息队列、
3、信号灯、以及共享内存区来说,有两个实现版本:POSIX的以及系统V的。Linux内核(内核2.4.18)支持POSIX信号灯、POSIX共享内存区以及POSIX消息队列,但对于主流Linux发行版本之一redhad8.0(内核2.4.18),还没有提供对POSIX进程间通信API的支持,不过应该只是时间上的事。因此,本文将主要介绍系统V消息队列及其相应API。在没有声明的情况下,以下讨论中指的都是系统V消息队列。一、消息队列基本概念1. 系统V消息队列是随内核持续的,只有在内核重起或者显示删除一个消息队列时,该消息队列才会真正被删除。因此系统中记录消息队列的数据结构(struct ipc_id
4、s msg_ids)位于内核中,系统中的所有消息队列都可以在结构msg_ids中找到访问入口。 2. 消息队列就是一个消息的链表。每个消息队列都有一个队列头,用结构struct msg_queue来描述(参见附录 2)。队列头中包含了该消息队列的大量信息,包括消息队列键值、用户ID、组ID、消息队列中消息数目等等,甚至记录了最近对消息队列读写进程的ID。读者可以访问这些信息,也可以设置其中的某些信息。 3. 下图说明了内核与消息队列是怎样建立起联系的:其中:struct ipc_ids msg_ids是内核中记录消息队列的全局数据结构;struct msg_queue是每个消息队列的队列头。
5、从上图可以看出,全局数据结构 struct ipc_ids msg_ids 可以访问到每个消息队列头的第一个成员:struct kern_ipc_perm;而每个struct kern_ipc_perm能够与具体的消息队列对应起来是因为在该结构中,有一个key_t类型成员key,而key则唯一确定一个消息队列。kern_ipc_perm结构如下:struct kern_ipc_perm /内核中记录消息队列的全局数据结构msg_ids能够访问到该结构; key_t key; /该键值则唯一对应一个消息队列 uid_t uid; gid_t gid;uid_t cuid;gid_t cgid;m
6、ode_t mode;unsigned long seq;二、操作消息队列对消息队列的操作无非有下面三种类型:1、 打开或创建消息队列消息队列的内核持续性要求每个消息队列都在系统范围内对应唯一的键值,所以,要获得一个消息队列的描述字,只需提供该消息队列的键值即可;注:消息队列描述字是由在系统范围内唯一的键值生成的,而键值可以看作对应系统内的一条路经。2、 读写操作消息读写操作非常简单,对开发人员来说,每个消息都类似如下的数据结构:struct msgbuflong mtype;char mtext1;mtype成员代表消息类型,从消息队列中读取消息的一个重要依据就是消息的类型;mtext是消息
7、内容,当然长度不一定为1。因此,对于发送消息来说,首先预置一个msgbuf缓冲区并写入消息类型和内容,调用相应的发送函数即可;对读取消息来说,首先分配这样一个msgbuf缓冲区,然后把消息读入该缓冲区即可。3、 获得或设置消息队列属性:消息队列的信息基本上都保存在消息队列头中,因此,可以分配一个类似于消息队列头的结构(struct msqid_ds,见附录 2),来返回消息队列的属性;同样可以设置该数据结构。消息队列API1、文件名到键值#include #include key_t ftok (char*pathname, char proj);它返回与路径pathname相对应的一个键值。
8、该函数不直接对消息队列操作,但在调用ipc(MSGGET,)或msgget()来获得消息队列描述字前,往往要调用该函数。典型的调用代码是: key=ftok(path_ptr, a); ipc_id=ipc(MSGGET, (int)key, flags,0,NULL,0); 2、linux为操作系统V进程间通信的三种方式(消息队列、信号灯、共享内存区)提供了一个统一的用户界面:int ipc(unsigned int call, int first, int second, int third, void *ptr, long fifth);第一个参数指明对IPC对象的操作方式,对消息队列而
9、言共有四种操作:MSGSND、MSGRCV、MSGGET以及MSGCTL,分别代表向消息队列发送消息、从消息队列读取消息、打开或创建消息队列、控制消息队列;first参数代表唯一的IPC对象;下面将介绍四种操作。 int ipc(MSGGET, int first, int second, int third, void *ptr, long fifth);与该操作对应的系统V调用为:int msgget( (key_t)first,second)。 int ipc(MSGCTL, int first, int second, int third, void *ptr, long fifth)
10、与该操作对应的系统V调用为:int msgctl( first,second, (struct msqid_ds*) ptr)。 int ipc(MSGSND, int first, int second, int third, void *ptr, long fifth);与该操作对应的系统V调用为:int msgsnd( first, (struct msgbuf*)ptr, second, third)。 int ipc(MSGRCV, int first, int second, int third, void *ptr, long fifth);与该操作对应的系统V调用为:int m
11、sgrcv( first,(struct msgbuf*)ptr, second, fifth,third), 注:本人不主张采用系统调用ipc(),而更倾向于采用系统V或者POSIX进程间通信API。原因如下: 虽然该系统调用提供了统一的用户界面,但正是由于这个特性,它的参数几乎不能给出特定的实际意义(如以first、second来命名参数),在一定程度上造成开发不便。 正如ipc手册所说的:ipc()是linux所特有的,编写程序时应注意程序的移植性问题; 该系统调用的实现不过是把系统V IPC函数进行了封装,没有任何效率上的优势; 系统V在IPC方面的API数量不多,形式也较简洁。 3.
12、系统V消息队列API系统V消息队列API共有四个,使用时需要包括几个头文件:#include #include #include 1)int msgget(key_t key, int msgflg)参数key是一个键值,由ftok获得;msgflg参数是一些标志位。该调用返回与健值key相对应的消息队列描述字。在以下两种情况下,该调用将创建一个新的消息队列: 如果没有消息队列与健值key相对应,并且msgflg中包含了IPC_CREAT标志位; key参数为IPC_PRIVATE; 参数msgflg可以为以下:IPC_CREAT、IPC_EXCL、IPC_NOWAIT或三者的或结果。调用返回
13、:成功返回消息队列描述字,否则返回-1。注:参数key设置成常数IPC_PRIVATE并不意味着其他进程不能访问该消息队列,只意味着即将创建新的消息队列。2)int msgrcv(int msqid, struct msgbuf *msgp, int msgsz, long msgtyp, int msgflg);该系统调用从msgid代表的消息队列中读取一个消息,并把消息存储在msgp指向的msgbuf结构中。msqid为消息队列描述字;消息返回后存储在msgp指向的地址,msgsz指定msgbuf的mtext成员的长度(即消息内容的长度),msgtyp为请求读取的消息类型;读消息标志msg
14、flg可以为以下几个常值的或: IPC_NOWAIT 如果没有满足条件的消息,调用立即返回,此时,errno=ENOMSG IPC_EXCEPT 与msgtyp0配合使用,返回队列中第一个类型不为msgtyp的消息 IPC_NOERROR 如果队列中满足条件的消息内容大于所请求的msgsz字节,则把该消息截断,截断部分将丢失。 msgrcv手册中详细给出了消息类型取不同值时(0; 0; =0),调用将返回消息队列中的哪个消息。msgrcv()解除阻塞的条件有三个:1. 消息队列中有了满足条件的消息; 2. msqid代表的消息队列被删除; 3. 调用msgrcv()的进程被信号中断; 调用返回
15、:成功返回读出消息的实际字节数,否则返回-1。3)int msgsnd(int msqid, struct msgbuf *msgp, int msgsz, int msgflg);向msgid代表的消息队列发送一个消息,即将发送的消息存储在msgp指向的msgbuf结构中,消息的大小由msgze指定。对发送消息来说,有意义的msgflg标志为IPC_NOWAIT,指明在消息队列没有足够空间容纳要发送的消息时,msgsnd是否等待。造成msgsnd()等待的条件有两种: 当前消息的大小与当前消息队列中的字节数之和超过了消息队列的总容量; 当前消息队列的消息数(单位个)不小于消息队列的总容量(单
16、位字节数),此时,虽然消息队列中的消息数目很多,但基本上都只有一个字节。 msgsnd()解除阻塞的条件有三个: 1. 不满足上述两个条件,即消息队列中有容纳该消息的空间; 2. msqid代表的消息队列被删除; 3. 调用msgsnd()的进程被信号中断; 调用返回:成功返回0,否则返回-1。4)int msgctl(int msqid, int cmd, struct msqid_ds *buf);该系统调用对由msqid标识的消息队列执行cmd操作,共有三种cmd操作:IPC_STAT、IPC_SET 、IPC_RMID。1. IPC_STAT:该命令用来获取消息队列信息,返回的信息存贮
17、在buf指向的msqid结构中; 2. IPC_SET:该命令用来设置消息队列的属性,要设置的属性存储在buf指向的msqid结构中;可设置属性包括:msg_perm.uid、msg_perm.gid、msg_perm.mode以及msg_qbytes,同时,也影响msg_ctime成员。 3. IPC_RMID:删除msqid标识的消息队列; 调用返回:成功返回0,否则返回-1。三、消息队列的限制每个消息队列的容量(所能容纳的字节数)都有限制,该值因系统不同而不同。在后面的应用实例中,输出了redhat 8.0的限制,结果参见附录 3。另一个限制是每个消息队列所能容纳的最大消息数:在redh
18、ad 8.0中,该限制是受消息队列容量制约的:消息个数要小于消息队列的容量(字节数)。注:上述两个限制是针对每个消息队列而言的,系统对消息队列的限制还有系统范围内的最大消息队列个数,以及整个系统范围内的最大消息数。一般来说,实际开发过程中不会超过这个限制。四、消息队列应用实例消息队列应用相对较简单,下面实例基本上覆盖了对消息队列的所有操作,同时,程序输出结果有助于加深对前面所讲的某些规则及消息队列限制的理解。#include #include #include void msg_stat(int,struct msqid_ds );main()int gflags,sflags,rflags;
19、key_t key;int msgid;int reval;struct msgsbuf int mtype; char mtext1; msg_sbuf;struct msgmbuf int mtype; char mtext10; msg_rbuf;struct msqid_ds msg_ginfo,msg_sinfo;char* msgpath=/unix/msgqueue;key=ftok(msgpath,a);gflags=IPC_CREAT|IPC_EXCL;msgid=msgget(key,gflags|00666);if(msgid=-1) printf(msg create
20、errorn); return;/创建一个消息队列后,输出消息队列缺省属性msg_stat(msgid,msg_ginfo);sflags=IPC_NOWAIT;msg_sbuf.mtype=10;msg_sbuf.mtext0=a;reval=msgsnd(msgid,&msg_sbuf,sizeof(msg_sbuf.mtext),sflags);if(reval=-1) printf(message send errorn);/发送一个消息后,输出消息队列属性msg_stat(msgid,msg_ginfo);rflags=IPC_NOWAIT|MSG_NOERROR;reval=msg
21、rcv(msgid,&msg_rbuf,4,10,rflags);if(reval=-1) printf(read msg errorn);else printf(read from msg queue %d bytesn,reval);/从消息队列中读出消息后,输出消息队列属性msg_stat(msgid,msg_ginfo);msg_sinfo.msg_perm.uid=8;/just a trymsg_sinfo.msg_perm.gid=8;/msg_sinfo.msg_qbytes=16388;/此处验证超级用户可以更改消息队列的缺省msg_qbytes/注意这里设置的值大于缺省值r
22、eval=msgctl(msgid,IPC_SET,&msg_sinfo);if(reval=-1) printf(msg set info errorn); return;msg_stat(msgid,msg_ginfo);/验证设置消息队列属性reval=msgctl(msgid,IPC_RMID,NULL);/删除消息队列if(reval=-1) printf(unlink msg queue errorn); return;void msg_stat(int msgid,struct msqid_ds msg_info)int reval;sleep(1);/只是为了后面输出时间的方便
23、reval=msgctl(msgid,IPC_STAT,&msg_info);if(reval=-1) printf(get msg info errorn); return;printf(n);printf(current number of bytes on queue is %dn,msg_info.msg_cbytes);printf(number of messages in queue is %dn,msg_info.msg_qnum);printf(max number of bytes on queue is %dn,msg_info.msg_qbytes);/每个消息队列的容
24、量(字节数)都有限制MSGMNB,值的大小因系统而异。在创建新的消息队列时,/msg_qbytes的缺省值就是MSGMNBprintf(pid of last msgsnd is %dn,msg_info.msg_lspid);printf(pid of last msgrcv is %dn,msg_info.msg_lrpid);printf(last msgsnd time is %s, ctime(&(msg_info.msg_stime);printf(last msgrcv time is %s, ctime(&(msg_info.msg_rtime);printf(last cha
25、nge time is %s, ctime(&(msg_info.msg_ctime);printf(msg uid is %dn,msg_info.msg_perm.uid);printf(msg gid is %dn,msg_info.msg_perm.gid);程序输出结果见附录 3。小结:消息队列与管道以及有名管道相比,具有更大的灵活性,首先,它提供有格式字节流,有利于减少开发人员的工作量;其次,消息具有类型,在实际应用中,可作为优先级使用。这两点是管道以及有名管道所不能比的。同样,消息队列可以在几个进程间复用,而不管这几个进程是否具有亲缘关系,这一点与有名管道很相似;但消息队列是随内
26、核持续的,与有名管道(随进程持续)相比,生命力更强,应用空间更大。附录 1:在参考文献1中,给出了IPC随进程持续、随内核持续以及随文件系统持续的定义:1. 随进程持续:IPC一直存在到打开IPC对象的最后一个进程关闭该对象为止。如管道和有名管道; 2. 随内核持续:IPC一直持续到内核重新自举或者显示删除该对象为止。如消息队列、信号灯以及共享内存等; 3. 随文件系统持续:IPC一直持续到显示删除该对象为止。 附录 2:结构msg_queue用来描述消息队列头,存在于系统空间:struct msg_queue struct kern_ipc_perm q_perm; time_t q_sti
27、me; /* last msgsnd time */ time_t q_rtime; /* last msgrcv time */ time_t q_ctime; /* last change time */ unsigned long q_cbytes; /* current number of bytes on queue */ unsigned long q_qnum; /* number of messages in queue */ unsigned long q_qbytes; /* max number of bytes on queue */ pid_t q_lspid; /*
28、 pid of last msgsnd */ pid_t q_lrpid; /* last receive pid */ struct list_head q_messages; struct list_head q_receivers; struct list_head q_senders;结构msqid_ds用来设置或返回消息队列的信息,存在于用户空间;struct msqid_ds struct ipc_perm msg_perm; struct msg *msg_first; /* first message on queue,unused */ struct msg *msg_las
29、t; /* last message in queue,unused */ _kernel_time_t msg_stime; /* last msgsnd time */ _kernel_time_t msg_rtime; /* last msgrcv time */ _kernel_time_t msg_ctime; /* last change time */ unsigned long msg_lcbytes; /* Reuse junk fields for 32 bit */ unsigned long msg_lqbytes; /* ditto */ unsigned short
30、 msg_cbytes; /* current number of bytes on queue */ unsigned short msg_qnum; /* number of messages in queue */ unsigned short msg_qbytes; /* max number of bytes on queue */ _kernel_ipc_pid_t msg_lspid; /* pid of last msgsnd */ _kernel_ipc_pid_t msg_lrpid; /* last receive pid */;/可以看出上述两个结构很相似。附录 3:消
31、息队列实例输出结果:current number of bytes on queue is 0number of messages in queue is 0max number of bytes on queue is 16384pid of last msgsnd is 0pid of last msgrcv is 0last msgsnd time is Thu Jan 1 08:00:00 1970last msgrcv time is Thu Jan 1 08:00:00 1970last change time is Sun Dec 29 18:28:20 2002msg uid
32、is 0msg gid is 0/上面刚刚创建一个新消息队列时的输出current number of bytes on queue is 1number of messages in queue is 1max number of bytes on queue is 16384pid of last msgsnd is 2510pid of last msgrcv is 0last msgsnd time is Sun Dec 29 18:28:21 2002last msgrcv time is Thu Jan 1 08:00:00 1970last change time is Sun
33、Dec 29 18:28:20 2002msg uid is 0msg gid is 0待添加的隐藏文字内容2read from msg queue 1 bytes/实际读出的字节数current number of bytes on queue is 0number of messages in queue is 0max number of bytes on queue is 16384 /每个消息队列最大容量(字节数)pid of last msgsnd is 2510pid of last msgrcv is 2510last msgsnd time is Sun Dec 29 18:
34、28:21 2002last msgrcv time is Sun Dec 29 18:28:22 2002last change time is Sun Dec 29 18:28:20 2002msg uid is 0msg gid is 0current number of bytes on queue is 0number of messages in queue is 0max number of bytes on queue is 16388 /可看出超级用户可修改消息队列最大容量pid of last msgsnd is 2510pid of last msgrcv is 2510
35、 /对操作消息队列进程的跟踪last msgsnd time is Sun Dec 29 18:28:21 2002last msgrcv time is Sun Dec 29 18:28:22 2002last change time is Sun Dec 29 18:28:23 2002 /msgctl()调用对msg_ctime有影响msg uid is 8msg gid is 8参考文献: UNIX网络编程第二卷:进程间通信,作者:W.Richard Stevens,译者:杨继张,清华大学出版社。对POSIX以及系统V消息队列都有阐述,对Linux环境下的程序开发有极大的启发意义。 l
36、inux内核源代码情景分析(上),毛德操、胡希明著,浙江大学出版社,给出了系统V消息队列相关的源代码分析。 msgget、msgsnd、msgrcv、msgctl手册Employment tribunals sort out disagreements between employers and employees.You may need to make a claim to an employment tribunal if: you dont agree with the disciplinary action your employer has taken against you yo
37、ur employer dismisses you and you think that you have been dismissed unfairly.For more information about dismissal and unfair dismissal, seeDismissal.You can make a claim to an employment tribunal, even if you haventappealedagainst the disciplinary action your employer has taken against you. However
38、, if you win your case, the tribunal may reduce any compensation awarded to you as a result of your failure to appeal.Remember that in most cases you must make an application to an employment tribunal within three months of the date when the event you are complaining about happened. If your applicat
39、ion is received after this time limit, the tribunal will not usually accept it.If you are worried about how the time limits apply to you, take advice from one of the organisations listed underFurther help.Employment tribunals are less formal than some other courts, but it is still a legal process an
40、d you will need to give evidence under an oath or affirmation.Most people find making a claim to an employment tribunal challenging. If you are thinking about making a claim to an employment tribunal, you should get help straight away from one of the organisations listed underFurther help.If you are
41、 being represented by a solicitor at the tribunal, they may ask you to sign an agreement where you pay their fee out of your compensation if you win the case. This is known as adamages-based agreement. In England and Wales, your solicitor cant charge you more than 35% of your compensation if you win
42、 the case.If you are thinking about signing up for a damages-based agreement, you should make sure youre clear about the terms of the agreement. It might be best to get advice from an experienced adviser, for example, at a Citizens Advice Bureau. To find your nearest CAB, including those that give a
43、dvice by e-mail, click onnearest CAB.For more information about making a claim to an employment tribunal, seeEmployment tribunals.The (lack of) air up there Watch mCayman Islands-based Webb, the head of Fifas anti-racism taskforce, is in London for the Football Associations 150th anniversary celebra
44、tions and will attend Citys Premier League match at Chelsea on Sunday.I am going to be at the match tomorrow and I have asked to meet Yaya Toure, he told BBC Sport.For me its about how he felt and I would like to speak to him first to find out what his experience was.Uefa hasopened disciplinary proc
45、eedings against CSKAfor the racist behaviour of their fans duringCitys 2-1 win.Michel Platini, president of European footballs governing body, has also ordered an immediate investigation into the referees actions.CSKA said they were surprised and disappointed by Toures complaint. In a statement the Russian side added: We found no racist insults from fans of CSKA.Age has reached the end of the beginning of a word. May be