青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

網(wǎng)絡(luò)服務(wù)器軟件開發(fā)/中間件開發(fā),關(guān)注ACE/ICE/boost

C++博客 首頁 新隨筆 聯(lián)系 聚合 管理
  152 Posts :: 3 Stories :: 172 Comments :: 0 Trackbacks

#

Linux環(huán)境進(jìn)程間通信(三)

消息隊(duì)列

developerWorks
?

未顯示需要 JavaScript 的文檔選項(xiàng)



級別: 初級

鄭彥興 (mlinux@163.com)國防科大計(jì)算機(jī)學(xué)院

2003 年 1 月 17 日

本系列文章中的前兩部分,我們探討管道及信號兩種通信機(jī)制,本文將深入第三部分,介紹系統(tǒng) V 消息隊(duì)列及其相應(yīng) API。

消息隊(duì)列(也叫做報(bào)文隊(duì)列)能夠克服早期unix通信機(jī)制的一些缺點(diǎn)。作為早期unix通信機(jī)制之一的信號能夠傳送的信息量有限,后來雖然POSIX 1003.1b在信號的實(shí)時(shí)性方面作了拓廣,使得信號在傳遞信息量方面有了相當(dāng)程度的改進(jìn),但是信號這種通信方式更像"即時(shí)"的通信方式,它要求接受信號的進(jìn)程在某個(gè)時(shí)間范圍內(nèi)對信號做出反應(yīng),因此該信號最多在接受信號進(jìn)程的生命周期內(nèi)才有意義,信號所傳遞的信息是接近于隨進(jìn)程持續(xù)的概念(process-persistent),見 附錄 1;管道及有名管道及有名管道則是典型的隨進(jìn)程持續(xù)IPC,并且,只能傳送無格式的字節(jié)流無疑會(huì)給應(yīng)用程序開發(fā)帶來不便,另外,它的緩沖區(qū)大小也受到限制。

消息隊(duì)列就是一個(gè)消息的鏈表。可以把消息看作一個(gè)記錄,具有特定的格式以及特定的優(yōu)先級。對消息隊(duì)列有寫權(quán)限的進(jìn)程可以向中按照一定的規(guī)則添加新消息;對消息隊(duì)列有讀權(quán)限的進(jìn)程則可以從消息隊(duì)列中讀走消息。消息隊(duì)列是隨內(nèi)核持續(xù)的(參見 附錄 1)。

目前主要有兩種類型的消息隊(duì)列:POSIX消息隊(duì)列以及系統(tǒng)V消息隊(duì)列,系統(tǒng)V消息隊(duì)列目前被大量使用。考慮到程序的可移植性,新開發(fā)的應(yīng)用程序應(yīng)盡量使用POSIX消息隊(duì)列。

在本系列專題的序(深刻理解Linux進(jìn)程間通信(IPC))中,提到對于消息隊(duì)列、信號燈、以及共享內(nèi)存區(qū)來說,有兩個(gè)實(shí)現(xiàn)版本:POSIX的以及系統(tǒng)V的。Linux內(nèi)核(內(nèi)核2.4.18)支持POSIX信號燈、POSIX共享內(nèi)存區(qū)以及POSIX消息隊(duì)列,但對于主流Linux發(fā)行版本之一redhad8.0(內(nèi)核2.4.18),還沒有提供對POSIX進(jìn)程間通信API的支持,不過應(yīng)該只是時(shí)間上的事。

因此,本文將主要介紹系統(tǒng)V消息隊(duì)列及其相應(yīng)API。 在沒有聲明的情況下,以下討論中指的都是系統(tǒng)V消息隊(duì)列。

一、消息隊(duì)列基本概念

  1. 系統(tǒng)V消息隊(duì)列是隨內(nèi)核持續(xù)的,只有在內(nèi)核重起或者顯示刪除一個(gè)消息隊(duì)列時(shí),該消息隊(duì)列才會(huì)真正被刪除。因此系統(tǒng)中記錄消息隊(duì)列的數(shù)據(jù)結(jié)構(gòu)(struct ipc_ids msg_ids)位于內(nèi)核中,系統(tǒng)中的所有消息隊(duì)列都可以在結(jié)構(gòu)msg_ids中找到訪問入口。
  2. 消息隊(duì)列就是一個(gè)消息的鏈表。每個(gè)消息隊(duì)列都有一個(gè)隊(duì)列頭,用結(jié)構(gòu)struct msg_queue來描述(參見 附錄 2)。隊(duì)列頭中包含了該消息隊(duì)列的大量信息,包括消息隊(duì)列鍵值、用戶ID、組ID、消息隊(duì)列中消息數(shù)目等等,甚至記錄了最近對消息隊(duì)列讀寫進(jìn)程的ID。讀者可以訪問這些信息,也可以設(shè)置其中的某些信息。
  3. 下圖說明了內(nèi)核與消息隊(duì)列是怎樣建立起聯(lián)系的:
    其中:struct ipc_ids msg_ids是內(nèi)核中記錄消息隊(duì)列的全局?jǐn)?shù)據(jù)結(jié)構(gòu);struct msg_queue是每個(gè)消息隊(duì)列的隊(duì)列頭。


從上圖可以看出,全局?jǐn)?shù)據(jù)結(jié)構(gòu) struct ipc_ids msg_ids 可以訪問到每個(gè)消息隊(duì)列頭的第一個(gè)成員:struct kern_ipc_perm;而每個(gè)struct kern_ipc_perm能夠與具體的消息隊(duì)列對應(yīng)起來是因?yàn)樵谠摻Y(jié)構(gòu)中,有一個(gè)key_t類型成員key,而key則唯一確定一個(gè)消息隊(duì)列。kern_ipc_perm結(jié)構(gòu)如下:

struct kern_ipc_perm{   //內(nèi)核中記錄消息隊(duì)列的全局?jǐn)?shù)據(jù)結(jié)構(gòu)msg_ids能夠訪問到該結(jié)構(gòu);
            key_t   key;    //該鍵值則唯一對應(yīng)一個(gè)消息隊(duì)列
            uid_t   uid;
            gid_t   gid;
uid_t   cuid;
gid_t   cgid;
mode_t  mode;
unsigned long seq;
}





回頁首


二、操作消息隊(duì)列

對消息隊(duì)列的操作無非有下面三種類型:

1、 打開或創(chuàng)建消息隊(duì)列
消息隊(duì)列的內(nèi)核持續(xù)性要求每個(gè)消息隊(duì)列都在系統(tǒng)范圍內(nèi)對應(yīng)唯一的鍵值,所以,要獲得一個(gè)消息隊(duì)列的描述字,只需提供該消息隊(duì)列的鍵值即可;

注:消息隊(duì)列描述字是由在系統(tǒng)范圍內(nèi)唯一的鍵值生成的,而鍵值可以看作對應(yīng)系統(tǒng)內(nèi)的一條路經(jīng)。

2、 讀寫操作

消息讀寫操作非常簡單,對開發(fā)人員來說,每個(gè)消息都類似如下的數(shù)據(jù)結(jié)構(gòu):

struct msgbuf{
long mtype;
char mtext[1];
};

mtype成員代表消息類型,從消息隊(duì)列中讀取消息的一個(gè)重要依據(jù)就是消息的類型;mtext是消息內(nèi)容,當(dāng)然長度不一定為1。因此,對于發(fā)送消息來說,首先預(yù)置一個(gè)msgbuf緩沖區(qū)并寫入消息類型和內(nèi)容,調(diào)用相應(yīng)的發(fā)送函數(shù)即可;對讀取消息來說,首先分配這樣一個(gè)msgbuf緩沖區(qū),然后把消息讀入該緩沖區(qū)即可。

3、 獲得或設(shè)置消息隊(duì)列屬性:

消息隊(duì)列的信息基本上都保存在消息隊(duì)列頭中,因此,可以分配一個(gè)類似于消息隊(duì)列頭的結(jié)構(gòu)(struct msqid_ds,見 附錄 2),來返回消息隊(duì)列的屬性;同樣可以設(shè)置該數(shù)據(jù)結(jié)構(gòu)。






消息隊(duì)列API

1、文件名到鍵值

#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok (char*pathname, char proj);


它返回與路徑pathname相對應(yīng)的一個(gè)鍵值。該函數(shù)不直接對消息隊(duì)列操作,但在調(diào)用ipc(MSGGET,…)或msgget()來獲得消息隊(duì)列描述字前,往往要調(diào)用該函數(shù)。典型的調(diào)用代碼是:

key=ftok(path_ptr, 'a');
    ipc_id=ipc(MSGGET, (int)key, flags,0,NULL,0);
    …


2、linux為操作系統(tǒng)V進(jìn)程間通信的三種方式(消息隊(duì)列、信號燈、共享內(nèi)存區(qū))提供了一個(gè)統(tǒng)一的用戶界面:
int ipc (unsigned int call, int first, int second, int third, void * ptr, long fifth);

第一個(gè)參數(shù)指明對IPC對象的操作方式,對消息隊(duì)列而言共有四種操作:MSGSND、MSGRCV、MSGGET以及MSGCTL,分別代表向消息隊(duì)列發(fā)送消息、從消息隊(duì)列讀取消息、打開或創(chuàng)建消息隊(duì)列、控制消息隊(duì)列;first參數(shù)代表唯一的IPC對象;下面將介紹四種操作。

  • int ipc( MSGGET, intfirst, intsecond, intthird, void*ptr, longfifth);
    與該操作對應(yīng)的系統(tǒng)V調(diào)用為:int msgget( (key_t)first,second)。
  • int ipc( MSGCTL, intfirst, intsecond, intthird, void*ptr, longfifth)
    與該操作對應(yīng)的系統(tǒng)V調(diào)用為:int msgctl( first,second, (struct msqid_ds*) ptr)。
  • int ipc( MSGSND, intfirst, intsecond, intthird, void*ptr, longfifth);
    與該操作對應(yīng)的系統(tǒng)V調(diào)用為:int msgsnd( first, (struct msgbuf*)ptr, second, third)。
  • int ipc( MSGRCV, intfirst, intsecond, intthird, void*ptr, longfifth);
    與該操作對應(yīng)的系統(tǒng)V調(diào)用為:int msgrcv( first,(struct msgbuf*)ptr, second, fifth,third),

注:本人不主張采用系統(tǒng)調(diào)用ipc(),而更傾向于采用系統(tǒng)V或者POSIX進(jìn)程間通信API。原因如下:

  • 雖然該系統(tǒng)調(diào)用提供了統(tǒng)一的用戶界面,但正是由于這個(gè)特性,它的參數(shù)幾乎不能給出特定的實(shí)際意義(如以first、second來命名參數(shù)),在一定程度上造成開發(fā)不便。
  • 正如ipc手冊所說的:ipc()是linux所特有的,編寫程序時(shí)應(yīng)注意程序的移植性問題;
  • 該系統(tǒng)調(diào)用的實(shí)現(xiàn)不過是把系統(tǒng)V IPC函數(shù)進(jìn)行了封裝,沒有任何效率上的優(yōu)勢;
  • 系統(tǒng)V在IPC方面的API數(shù)量不多,形式也較簡潔。

3.系統(tǒng)V消息隊(duì)列API
系統(tǒng)V消息隊(duì)列API共有四個(gè),使用時(shí)需要包括幾個(gè)頭文件:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>


1)int msgget(key_t key, int msgflg)

參數(shù)key是一個(gè)鍵值,由ftok獲得;msgflg參數(shù)是一些標(biāo)志位。該調(diào)用返回與健值key相對應(yīng)的消息隊(duì)列描述字。

在以下兩種情況下,該調(diào)用將創(chuàng)建一個(gè)新的消息隊(duì)列:

  • 如果沒有消息隊(duì)列與健值key相對應(yīng),并且msgflg中包含了IPC_CREAT標(biāo)志位;
  • key參數(shù)為IPC_PRIVATE;

參數(shù)msgflg可以為以下:IPC_CREAT、IPC_EXCL、IPC_NOWAIT或三者的或結(jié)果。

調(diào)用返回:成功返回消息隊(duì)列描述字,否則返回-1。

注:參數(shù)key設(shè)置成常數(shù)IPC_PRIVATE并不意味著其他進(jìn)程不能訪問該消息隊(duì)列,只意味著即將創(chuàng)建新的消息隊(duì)列。

2)int msgrcv(int msqid, struct msgbuf *msgp, int msgsz, long msgtyp, int msgflg);
該系統(tǒng)調(diào)用從msgid代表的消息隊(duì)列中讀取一個(gè)消息,并把消息存儲(chǔ)在msgp指向的msgbuf結(jié)構(gòu)中。

msqid為消息隊(duì)列描述字;消息返回后存儲(chǔ)在msgp指向的地址,msgsz指定msgbuf的mtext成員的長度(即消息內(nèi)容的長度),msgtyp為請求讀取的消息類型;讀消息標(biāo)志msgflg可以為以下幾個(gè)常值的或:

  • IPC_NOWAIT 如果沒有滿足條件的消息,調(diào)用立即返回,此時(shí),errno=ENOMSG
  • IPC_EXCEPT 與msgtyp>0配合使用,返回隊(duì)列中第一個(gè)類型不為msgtyp的消息
  • IPC_NOERROR 如果隊(duì)列中滿足條件的消息內(nèi)容大于所請求的msgsz字節(jié),則把該消息截?cái)啵財(cái)嗖糠謱G失。

msgrcv手冊中詳細(xì)給出了消息類型取不同值時(shí)(>0; <0; =0),調(diào)用將返回消息隊(duì)列中的哪個(gè)消息。

msgrcv()解除阻塞的條件有三個(gè):

  1. 消息隊(duì)列中有了滿足條件的消息;
  2. msqid代表的消息隊(duì)列被刪除;
  3. 調(diào)用msgrcv()的進(jìn)程被信號中斷;

調(diào)用返回:成功返回讀出消息的實(shí)際字節(jié)數(shù),否則返回-1。

3)int msgsnd(int msqid, struct msgbuf *msgp, int msgsz, int msgflg);
向msgid代表的消息隊(duì)列發(fā)送一個(gè)消息,即將發(fā)送的消息存儲(chǔ)在msgp指向的msgbuf結(jié)構(gòu)中,消息的大小由msgze指定。

對發(fā)送消息來說,有意義的msgflg標(biāo)志為IPC_NOWAIT,指明在消息隊(duì)列沒有足夠空間容納要發(fā)送的消息時(shí),msgsnd是否等待。造成msgsnd()等待的條件有兩種:

  • 當(dāng)前消息的大小與當(dāng)前消息隊(duì)列中的字節(jié)數(shù)之和超過了消息隊(duì)列的總?cè)萘浚?
  • 當(dāng)前消息隊(duì)列的消息數(shù)(單位"個(gè)")不小于消息隊(duì)列的總?cè)萘浚▎挝?字節(jié)數(shù)"),此時(shí),雖然消息隊(duì)列中的消息數(shù)目很多,但基本上都只有一個(gè)字節(jié)。

msgsnd()解除阻塞的條件有三個(gè):
  1. 不滿足上述兩個(gè)條件,即消息隊(duì)列中有容納該消息的空間;
  2. msqid代表的消息隊(duì)列被刪除;
  3. 調(diào)用msgsnd()的進(jìn)程被信號中斷;

調(diào)用返回:成功返回0,否則返回-1。

4)int msgctl(int msqid, int cmd, struct msqid_ds *buf);
該系統(tǒng)調(diào)用對由msqid標(biāo)識(shí)的消息隊(duì)列執(zhí)行cmd操作,共有三種cmd操作:IPC_STAT、IPC_SET 、IPC_RMID。

  1. IPC_STAT:該命令用來獲取消息隊(duì)列信息,返回的信息存貯在buf指向的msqid結(jié)構(gòu)中;
  2. IPC_SET:該命令用來設(shè)置消息隊(duì)列的屬性,要設(shè)置的屬性存儲(chǔ)在buf指向的msqid結(jié)構(gòu)中;可設(shè)置屬性包括:msg_perm.uid、msg_perm.gid、msg_perm.mode以及msg_qbytes,同時(shí),也影響msg_ctime成員。
  3. IPC_RMID:刪除msqid標(biāo)識(shí)的消息隊(duì)列;

調(diào)用返回:成功返回0,否則返回-1。





回頁首


三、消息隊(duì)列的限制

每個(gè)消息隊(duì)列的容量(所能容納的字節(jié)數(shù))都有限制,該值因系統(tǒng)不同而不同。在后面的應(yīng)用實(shí)例中,輸出了redhat 8.0的限制,結(jié)果參見 附錄 3

另一個(gè)限制是每個(gè)消息隊(duì)列所能容納的最大消息數(shù):在redhad 8.0中,該限制是受消息隊(duì)列容量制約的:消息個(gè)數(shù)要小于消息隊(duì)列的容量(字節(jié)數(shù))。

注:上述兩個(gè)限制是針對每個(gè)消息隊(duì)列而言的,系統(tǒng)對消息隊(duì)列的限制還有系統(tǒng)范圍內(nèi)的最大消息隊(duì)列個(gè)數(shù),以及整個(gè)系統(tǒng)范圍內(nèi)的最大消息數(shù)。一般來說,實(shí)際開發(fā)過程中不會(huì)超過這個(gè)限制。





回頁首


四、消息隊(duì)列應(yīng)用實(shí)例

消息隊(duì)列應(yīng)用相對較簡單,下面實(shí)例基本上覆蓋了對消息隊(duì)列的所有操作,同時(shí),程序輸出結(jié)果有助于加深對前面所講的某些規(guī)則及消息隊(duì)列限制的理解。

#include <sys/types.h>
#include <sys/msg.h>
#include <unistd.h>
void msg_stat(int,struct msqid_ds );
main()
{
int gflags,sflags,rflags;
key_t key;
int msgid;
int reval;
struct msgsbuf{
        int mtype;
        char mtext[1];
    }msg_sbuf;
struct msgmbuf
    {
    int mtype;
    char mtext[10];
    }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 error\n");
    return;
}
//創(chuàng)建一個(gè)消息隊(duì)列后,輸出消息隊(duì)列缺省屬性
msg_stat(msgid,msg_ginfo);
sflags=IPC_NOWAIT;
msg_sbuf.mtype=10;
msg_sbuf.mtext[0]='a';
reval=msgsnd(msgid,&msg_sbuf,sizeof(msg_sbuf.mtext),sflags);
if(reval==-1)
{
    printf("message send error\n");
}
//發(fā)送一個(gè)消息后,輸出消息隊(duì)列屬性
msg_stat(msgid,msg_ginfo);
rflags=IPC_NOWAIT|MSG_NOERROR;
reval=msgrcv(msgid,&msg_rbuf,4,10,rflags);
if(reval==-1)
    printf("read msg error\n");
else
    printf("read from msg queue %d bytes\n",reval);
//從消息隊(duì)列中讀出消息后,輸出消息隊(duì)列屬性
msg_stat(msgid,msg_ginfo);
msg_sinfo.msg_perm.uid=8;//just a try
msg_sinfo.msg_perm.gid=8;//
msg_sinfo.msg_qbytes=16388;
//此處驗(yàn)證超級用戶可以更改消息隊(duì)列的缺省msg_qbytes
//注意這里設(shè)置的值大于缺省值
reval=msgctl(msgid,IPC_SET,&msg_sinfo);
if(reval==-1)
{
    printf("msg set info error\n");
    return;
}
msg_stat(msgid,msg_ginfo);
//驗(yàn)證設(shè)置消息隊(duì)列屬性
reval=msgctl(msgid,IPC_RMID,NULL);//刪除消息隊(duì)列
if(reval==-1)
{
    printf("unlink msg queue error\n");
    return;
}
}
void msg_stat(int msgid,struct msqid_ds msg_info)
{
int reval;
sleep(1);//只是為了后面輸出時(shí)間的方便
reval=msgctl(msgid,IPC_STAT,&msg_info);
if(reval==-1)
{
    printf("get msg info error\n");
    return;
}
printf("\n");
printf("current number of bytes on queue is %d\n",msg_info.msg_cbytes);
printf("number of messages in queue is %d\n",msg_info.msg_qnum);
printf("max number of bytes on queue is %d\n",msg_info.msg_qbytes);
//每個(gè)消息隊(duì)列的容量(字節(jié)數(shù))都有限制MSGMNB,值的大小因系統(tǒng)而異。在創(chuàng)建新的消息隊(duì)列時(shí),//msg_qbytes的缺省值就是MSGMNB
printf("pid of last msgsnd is %d\n",msg_info.msg_lspid);
printf("pid of last msgrcv is %d\n",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 change time is %s", ctime(&(msg_info.msg_ctime)));
printf("msg uid is %d\n",msg_info.msg_perm.uid);
printf("msg gid is %d\n",msg_info.msg_perm.gid);
}

程序輸出結(jié)果見 附錄 3




回頁首


小結(jié):

消息隊(duì)列與管道以及有名管道相比,具有更大的靈活性,首先,它提供有格式字節(jié)流,有利于減少開發(fā)人員的工作量;其次,消息具有類型,在實(shí)際應(yīng)用中,可作為優(yōu)先級使用。這兩點(diǎn)是管道以及有名管道所不能比的。同樣,消息隊(duì)列可以在幾個(gè)進(jìn)程間復(fù)用,而不管這幾個(gè)進(jìn)程是否具有親緣關(guān)系,這一點(diǎn)與有名管道很相似;但消息隊(duì)列是隨內(nèi)核持續(xù)的,與有名管道(隨進(jìn)程持續(xù))相比,生命力更強(qiáng),應(yīng)用空間更大。

附錄 1在參考文獻(xiàn)[1]中,給出了IPC隨進(jìn)程持續(xù)、隨內(nèi)核持續(xù)以及隨文件系統(tǒng)持續(xù)的定義:

  1. 隨進(jìn)程持續(xù):IPC一直存在到打開IPC對象的最后一個(gè)進(jìn)程關(guān)閉該對象為止。如管道和有名管道;
  2. 隨內(nèi)核持續(xù):IPC一直持續(xù)到內(nèi)核重新自舉或者顯示刪除該對象為止。如消息隊(duì)列、信號燈以及共享內(nèi)存等;
  3. 隨文件系統(tǒng)持續(xù):IPC一直持續(xù)到顯示刪除該對象為止。

附錄 2
結(jié)構(gòu)msg_queue用來描述消息隊(duì)列頭,存在于系統(tǒng)空間:

struct msg_queue {
    struct kern_ipc_perm q_perm;
    time_t q_stime;         /* 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;          /* 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;
};


結(jié)構(gòu)msqid_ds用來設(shè)置或返回消息隊(duì)列的信息,存在于用戶空間;

struct msqid_ds {
    struct ipc_perm msg_perm;
    struct msg *msg_first;      /* first message on queue,unused  */
    struct msg *msg_last;       /* 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 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 */
};

//可以看出上述兩個(gè)結(jié)構(gòu)很相似。

附錄 3消息隊(duì)列實(shí)例輸出結(jié)果:

current number of bytes on queue is 0
number of messages in queue is 0
max number of bytes on queue is 16384
pid of last msgsnd is 0
pid of last msgrcv is 0
last msgsnd time is Thu Jan  1 08:00:00 1970
last msgrcv time is Thu Jan  1 08:00:00 1970
last change time is Sun Dec 29 18:28:20 2002
msg uid is 0
msg gid is 0
//上面剛剛創(chuàng)建一個(gè)新消息隊(duì)列時(shí)的輸出
current number of bytes on queue is 1
number of messages in queue is 1
max number of bytes on queue is 16384
pid of last msgsnd is 2510
pid of last msgrcv is 0
last msgsnd time is Sun Dec 29 18:28:21 2002
last msgrcv time is Thu Jan  1 08:00:00 1970
last change time is Sun Dec 29 18:28:20 2002
msg uid is 0
msg gid is 0
read from msg queue 1 bytes
//實(shí)際讀出的字節(jié)數(shù)
current number of bytes on queue is 0
number of messages in queue is 0
max number of bytes on queue is 16384   //每個(gè)消息隊(duì)列最大容量(字節(jié)數(shù))
pid of last msgsnd is 2510
pid of last msgrcv is 2510
last msgsnd time is Sun Dec 29 18:28:21 2002
last msgrcv time is Sun Dec 29 18:28:22 2002
last change time is Sun Dec 29 18:28:20 2002
msg uid is 0
msg gid is 0
current number of bytes on queue is 0
number of messages in queue is 0
max number of bytes on queue is 16388   //可看出超級用戶可修改消息隊(duì)列最大容量
pid of last msgsnd is 2510
pid of last msgrcv is 2510  //對操作消息隊(duì)列進(jìn)程的跟蹤
last msgsnd time is Sun Dec 29 18:28:21 2002
last msgrcv time is Sun Dec 29 18:28:22 2002
last change time is Sun Dec 29 18:28:23 2002    //msgctl()調(diào)用對msg_ctime有影響
msg uid is 8
msg gid is 8



參考資料

  • UNIX網(wǎng)絡(luò)編程第二卷:進(jìn)程間通信,作者:W.Richard Stevens,譯者:楊繼張,清華大學(xué)出版社。對POSIX以及系統(tǒng)V消息隊(duì)列都有闡述,對Linux環(huán)境下的程序開發(fā)有極大的啟發(fā)意義。

  • linux內(nèi)核源代碼情景分析(上),毛德操、胡希明著,浙江大學(xué)出版社,給出了系統(tǒng)V消息隊(duì)列相關(guān)的源代碼分析。

  • http://www.fanqiang.com/a4/b2/20010508/113315.html,主要闡述linux下對文件的操作,詳細(xì)介紹了對文件的存取權(quán)限位,對IPC對象的存取權(quán)限同樣具有很好的借鑒意義。

  • msgget、msgsnd、msgrcv、msgctl手冊


關(guān)于作者

鄭彥興,男,現(xiàn)攻讀國防科大計(jì)算機(jī)學(xué)院網(wǎng)絡(luò)方向博士學(xué)位。您可以通過電子郵件 mlinux@163.com和他聯(lián)系。

posted @ 2007-03-27 16:26 true 閱讀(442) | 評論 (0)編輯 收藏

ldconfig幾個(gè)需要注意的地方

1. 往/lib和/usr/lib里面加?xùn)|西,是不用修改/etc/ld.so.conf的,但是完了之后要調(diào)一下ldconfig,不然這個(gè)library會(huì)找不到
2. 想往上面兩個(gè)目錄以外加?xùn)|西的時(shí)候,一定要修改/etc/ld.so.conf,然后再調(diào)用ldconfig,不然也會(huì)找不到
比如安裝了一個(gè)mysql到/usr/local/mysql,mysql有一大堆library在/usr/local/mysql/lib下面,這時(shí)就需要在/etc/ld.so.conf下面加一行/usr/local/mysql/lib,保存過后ldconfig一下,新的library才能在程序運(yùn)行時(shí)被找到。
3. 如果想在這兩個(gè)目錄以外放lib,但是又不想在/etc/ld.so.conf中加?xùn)|西(或者是沒有權(quán)限加?xùn)|西)。那也可以,就是export一個(gè)全局變量LD_LIBRARY_PATH,然后運(yùn)行程序的時(shí)候就會(huì)去這個(gè)目錄中找library。一般來講這只是一種臨時(shí)的解決方案,在沒有權(quán)限或臨時(shí)需要的時(shí)候使用。
4. ldconfig做的這些東西都與運(yùn)行程序時(shí)有關(guān),跟編譯時(shí)一點(diǎn)關(guān)系都沒有。編譯的時(shí)候還是該加-L就得加,不要混淆了。
5. 總之,就是不管做了什么關(guān)于library的變動(dòng)后,最好都ldconfig一下,不然會(huì)出現(xiàn)一些意想不到的結(jié)果。不會(huì)花太多的時(shí)間,但是會(huì)省很多的事。
posted @ 2007-03-27 08:53 true 閱讀(4344) | 評論 (4)編輯 收藏

std::cout << "0x" << std::hex << 255 << std::endl;
輸出? 0xff;
posted @ 2007-03-19 17:56 true 閱讀(2077) | 評論 (0)編輯 收藏

Xerces C++ 學(xué)習(xí)筆記

Xerces C++ 學(xué)習(xí)筆記

1. ? 初始化平臺(tái):

XMLPlatformUtils::Initialze()

2. ? 銷毀平臺(tái):

XMLPlatformUtils::Terminate();

3. ? 加載分析報(bào)文

XercesDOMParser? *parser = new XercesDOMParser();

Parser->parse( 參數(shù) ); 這個(gè)參數(shù)可以直接是文件名,也可以是內(nèi)存的數(shù)據(jù),具體的查看 API

4. ? 數(shù)據(jù)格式轉(zhuǎn)換

XMLString.transcode() 這個(gè)方法有多個(gè)重載,既可以把 XMLCh * 格式的數(shù)據(jù)轉(zhuǎn)換成 char * 的數(shù)據(jù),也可以把 char * 的數(shù)據(jù)轉(zhuǎn)換成 XMLCh *, 我們也可以利用這個(gè)函數(shù)來初始化 XMLCh 格式類型的數(shù)組,比如 XMLCh temp[100];

??????????????????????????? XMLString.transcode(“LS”,temp,99);// 用“ LS ”初始化 temp 這個(gè)數(shù)組的內(nèi)容。

5. ? 分析報(bào)文后,如何得到文檔的樹型結(jié)構(gòu)

DOMDocument *xmlDoc = parser->getDocument();

6. ? 得到樹型結(jié)構(gòu)后,如何得到第一個(gè)根節(jié)點(diǎn)

DOMElement *root = xmlDoc->getDocumentElement();

7. ? 如何遍歷文檔的結(jié)構(gòu)

遍歷一個(gè)樹型的文檔結(jié)構(gòu)有三種方法。

1 .使用 DOMNodeIterator

DOMNodeIterator *iterator = xmlDoc->createNodeIterator(root, DOMNodeFilter::SHOW_TEXT,? NULL, true);

for ( DOMNode * current = (DOMNode *)iterator->nextNode(); current != 0; current = (DOMNode *)iterator->nextNode() )

{

??????? string strValue = XMLString::transcode(current->getNodeValue());

??????? std::cout <<strValue<<endl;

?}// 以上就可以把 xml 文檔中的屬性為 NODETEXT 節(jié)點(diǎn)的內(nèi)容給打印出來。

2 .使用 DOMTreeWalker

D OMTreeWalker *walker =??xmlDoc->createTreeWalker(root, DOMNodeFilter::SHOW_TEXT, NULL, true);

f or (DOMNode *current = walker->nextNode(); current != 0; current = walker->nextNode() )? ?

?{

? ?? ?char *strValue = XMLString::transcode( current->getNodeValue() );? ? ?std::cout <<strValue;
? ?? ?? ?? ?XMLString::release(&strValue);? ?

}

3 .使用子節(jié)點(diǎn)直接遍歷樹型結(jié)構(gòu)

?? DOMNode *n = (DOMNode*)xmlDoc->getDocumentElement();

// 下面開始遍歷這個(gè)樹的結(jié)構(gòu)

if(n)

{

??????? if (n->getNodeType() == DOMNode::ELEMENT_NODE)

??????? {

?????????????? ?DOMNodeList* nodeList = n->getChildNodes();

?????????????? ?unsigned int nListLen = nodeList->getLength();

?????????????? ?for (unsigned int i=0; i<nListLen; ++i)

?????????????? ?{

????????????????????? DOMNode* nodeTemp = nodeList->item(i);

????????????????????? if (nodeTemp->getNodeType() == DOMNode::ELEMENT_NODE)

????????????????????? {

???????????????????????????? for (DOMNode* node1=nodeTemp->getFirstChild(); node1!=0; node1=node1->getNextSibling())

???????????????????????????? {

??????????????????????????????????? char* name = XMLString::transcode(node1->getNodeName());

??????????????????????????????????? string strTemp = name;

??????????????????????????????????? if (strTemp == "name")?? // 這個(gè)就是跟 xml 文檔中 name 節(jié)點(diǎn)匹配

??????????????????????????????????? {

??????? char* myname=XMLString::transcode(node1->getFirstChild()->getNodeValue());

?????????????????????????????????????????? cout<<myname<<endl;

??????????????????????????????????? }

???????????????????????????? }

????????????????????? }

????????????????????? continue;

?????????????? }

??????? }

}

8. ? 如何添加子節(jié)點(diǎn)。

??? //Add new (empty) Element to the root element
???? DOM_Element parentNode = …;// parent is known
???? DOM_Element prodElem = doc->createElement (tagName);
???? parentNode->appendChild (prodElem);

9. ? 在加載 xml 文件分析前,有兩種加載方式:一種直接通過文件加載,一種是通過內(nèi)存加載

1. ? String xmlfile = “a.xml”;

Parser->parse(xmlfile.c_str());

??????? 2.MemBufInputSource ? 這個(gè)類處理內(nèi)存的數(shù)據(jù),然后利用 parser->parse(*men)//men MemBufInputSource ? 的實(shí)例指針對象。

?

10. ????????????? 在加載分析報(bào)文前,可以設(shè)置 XercesDOMParser 的一些屬性。

??????? P arser->setValidationScheme( XercesDOMParser::Val_Auto );
? ??? ????Parser->setDoNamespaces( false );
? ? ?????Parser->setDoSchema( false );
? ? ?????Parser->setLoadExternalDTD( false );

??????? // 這個(gè)部分的代碼可以參照類庫自帶的例子。

11. ????????????? 我們也可以設(shè)置錯(cuò)誤的處理代碼。

??????? 這部分可以查看類庫下的 DOMPrint 例子。 DOMError ,DOMErrorHandler

12. ????????????? 也可以設(shè)置分析過濾的屬性

???????? DOMNodeFilter? 這個(gè)類能起到這個(gè)效果

13. ????????????? 要美化文檔的格式化輸出,我們可以使用

??????? XMLFormatter??? XMLFormatTarget? , 用的時(shí)候可以查看。

14????? 我們也可以通過 DOMImplementationLs , DOMImplementation 前者是后者的基類。通過這個(gè)類我們可以創(chuàng)建 DOMBuilder parser )或 DOMWriter (序列,既可以寫到屏幕,也可以寫到文件)

posted @ 2007-03-15 15:54 true 閱讀(4801) | 評論 (4)編輯 收藏

僅列出標(biāo)題
共15頁: First 7 8 9 10 11 12 13 14 15 
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产精品一二| 欧美一级电影久久| 亚洲视频你懂的| 欧美日韩国产片| 久久免费国产精品| 亚洲人成7777| 亚洲综合精品一区二区| 国产美女精品| 欧美激情第8页| 国产精品影视天天线| 蘑菇福利视频一区播放| 影音先锋久久| 欧美1区免费| 国产综合色在线| 欧美在线观看视频| 亚洲一区在线视频| 欧美一站二站| 国语对白精品一区二区| 欧美亚洲免费高清在线观看| 亚洲国产一区二区a毛片| 国产精品久久九九| 国产亚洲欧洲997久久综合| 一区二区三区免费观看| 亚洲免费一在线| 一区二区三区四区精品| 亚洲欧美日韩在线不卡| 免费毛片一区二区三区久久久| 欧美一区二区成人| 国产日本欧美在线观看| 亚洲图片你懂的| 亚洲一区国产一区| 一色屋精品视频在线观看网站| 一区二区免费在线观看| 一区二区三区日韩在线观看| 欧美成人精品| 欧美激情中文字幕一区二区| 国产亚洲欧美一区在线观看| 亚洲一区二区三区在线观看视频| 欧美午夜电影一区| 亚洲美女网站| 亚洲视频一区| 欧美日韩不卡合集视频| 亚洲一二三区精品| 欧美一二区视频| 国产一区二区三区四区五区美女 | 亚洲欧洲在线看| 欧美日韩国产成人在线观看| 久久久久成人精品免费播放动漫| 在线亚洲欧美视频| 国产精品日本精品| 久久精品一区二区三区中文字幕| 美女在线一区二区| 精品成人国产| 亚洲美女av在线播放| 久久久www成人免费无遮挡大片| 国内视频一区| 国产精品av久久久久久麻豆网| 性8sex亚洲区入口| 亚洲国产激情| 亚洲国产日韩一区| 亚洲国产精品一区二区三区| 欧美日本久久| 久久综合给合久久狠狠色| 欧美另类女人| 久久婷婷国产综合国色天香| 欧美精品乱码久久久久久按摩| 亚洲自拍偷拍视频| 嫩草成人www欧美| 久久久久久久久岛国免费| 欧美日韩爆操| 亚洲韩国一区二区三区| 韩日在线一区| 亚洲午夜精品久久| 日韩一级免费| 久久夜色精品国产| 久久精品久久综合| 国产精品久久国产精品99gif| 欧美激情网友自拍| 在线播放日韩| 久久精品国产第一区二区三区最新章节 | 国产欧美日本在线| 亚洲精品国久久99热| 影音先锋欧美精品| 亚洲欧美日韩成人高清在线一区| 日韩视频一区二区三区| 老司机67194精品线观看| 久久免费精品视频| 国产午夜亚洲精品羞羞网站| 一区二区日韩| 一区二区三区久久久| 欧美激情免费观看| 亚洲人www| 一区二区免费在线观看| 欧美激情视频一区二区三区在线播放 | 欧美高清视频| 在线欧美不卡| 另类春色校园亚洲| 老牛国产精品一区的观看方式| 国产亚洲欧美日韩一区二区| 亚洲欧美日韩国产综合| 欧美一区二区精品久久911| 国产精品麻豆欧美日韩ww| 免费日韩av| 午夜一区二区三视频在线观看 | 亚洲免费在线观看视频| 欧美日韩综合视频| 亚洲精品一区二区三区蜜桃久 | 欧美日韩国产经典色站一区二区三区| 亚洲高清电影| av成人毛片| 欧美丝袜一区二区| 亚洲曰本av电影| 久久久精品2019中文字幕神马| 国产日韩在线看| 久久全球大尺度高清视频| 欧美激情精品久久久久久变态| 日韩一本二本av| 国产精品日韩精品欧美在线| 午夜激情综合网| 另类亚洲自拍| 91久久精品美女高潮| 欧美精品二区| 亚洲午夜精品一区二区三区他趣| 亚洲少妇一区| 国产午夜久久久久| 欧美久久电影| 欧美一区视频| 亚洲片在线资源| 久久久亚洲精品一区二区三区 | 亚洲人成小说网站色在线| 蜜桃av噜噜一区| 日韩视频在线免费| 亚洲欧美伊人| 在线观看91精品国产麻豆| 欧美成人午夜77777| 99亚洲一区二区| 久久精品91| 亚洲乱码国产乱码精品精98午夜| 欧美日韩国产限制| 在线综合+亚洲+欧美中文字幕| 久久婷婷影院| 亚洲作爱视频| 精品不卡视频| 国产精品系列在线| 欧美激情第二页| 性久久久久久久| 亚洲美女黄色| 欧美α欧美αv大片| 亚洲视频 欧洲视频| 激情五月***国产精品| 国产精品电影在线观看| 久久免费国产精品1| 亚洲欧美一区二区三区极速播放 | 亚洲毛片视频| 久久天天综合| 欧美一区二区免费观在线| 亚洲高清免费视频| 国产一区二区三区久久 | 欧美成年人视频| 亚洲伊人色欲综合网| 亚洲精品视频一区| 亚洲美女电影在线| 亚洲图片欧美日产| 久久亚洲视频| 亚洲一区二区在线| 亚洲国产1区| 久久久视频精品| 性欧美大战久久久久久久免费观看 | 国产精品美女久久久久久免费| 欧美va天堂在线| 欧美一区=区| 亚洲图片在区色| 日韩视频专区| 亚洲激情专区| 91久久中文字幕| 亚洲黄色影片| 欧美激情综合| 久久综合中文字幕| 久久国产精品久久国产精品| 性高湖久久久久久久久| 校园激情久久| 欧美一区二区三区男人的天堂| 亚洲欧美国产另类| 先锋影音国产精品| 欧美中文字幕在线观看| 欧美一区免费视频| 久久久久久久综合色一本| 久久精品一区二区三区中文字幕| 欧美在线视频二区| 久久蜜桃av一区精品变态类天堂| 欧美一区二区在线免费播放| 午夜影院日韩| 午夜一区在线| 久久精品国产亚洲一区二区| 欧美一区二区| 久久综合久久久久88| 欧美成人综合| 欧美性猛交xxxx免费看久久久 | 亚洲第一中文字幕| 亚洲精品国产精品国自产观看|