• <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>
            posts - 15,comments - 21,trackbacks - 0
            進(jìn)程間通信方式包括了管道,消息隊(duì)列,F(xiàn)IFO,共享內(nèi)存,而共享內(nèi)存是其中效率最高的。下圖解釋了其效率最高的原因(圖片截取自《UNIX網(wǎng)絡(luò)編程》)

            我們可以看到上面的拷貝次數(shù)是4次,下面則是2次。
            接下來(lái)我們看看使用,其實(shí)網(wǎng)上和書(shū)上都有了很多資料,API就是那么幾個(gè)。
            int shmget(key_t key,size_t size,int shmflag)。
            key:可以指定IPC_PRIVATE,那么系統(tǒng)會(huì)為你創(chuàng)建一個(gè)key,并且返回一個(gè)id號(hào)。也可以通過(guò)ftok函數(shù)生成一個(gè)key(不了解ftok的童鞋可以動(dòng)手man一下)。那么為什么要一個(gè)IPC既要一個(gè)key又有一個(gè)ID呢。這里我覺(jué)得是為了方便其他進(jìn)程訪(fǎng)問(wèn)。進(jìn)程A創(chuàng)建了一個(gè)共享內(nèi)存,內(nèi)核為其分配一個(gè)ID,這個(gè)時(shí)候進(jìn)程B想要訪(fǎng)問(wèn),他怎么獲取這個(gè)ID(難道需要A把ID發(fā)送給B??)。但是我們用一個(gè)key就很方便了。事先A,B進(jìn)程都知道這個(gè)key,那么A創(chuàng)建了,B就可以通過(guò)事先知道key找到這塊內(nèi)存。
            size:共享內(nèi)存的大小。如果是獲得一塊內(nèi)存,則該值應(yīng)該為0。
            shmflag:讀寫(xiě)權(quán)限的組合,可以與IPC_CREAT和IPC_EXCL按位或。當(dāng)指定一個(gè)key時(shí),IPC_CREAT和IPC_EXCL配合使用可以在存在該key的共享內(nèi)存時(shí)返回-1。
            當(dāng)該函數(shù)調(diào)用成功后,返回一個(gè)系統(tǒng)分配的共享內(nèi)存,并且size大小的字節(jié)被初始化為0

            void *shmat(int shmid, const void *shmaddr, int shmflg)
            有了一塊共享內(nèi)存后,進(jìn)程需要映射該內(nèi)存到進(jìn)程的地址空間。這個(gè)函數(shù)就是作用。
            shmid就是之前獲得ID,shmaddr如果指定了,就會(huì)配合shmflg確定映射地址,不過(guò)一般都不這么干的。返回值就是獲得的地址,你可以往里面寫(xiě)你或者讀你需要的數(shù)據(jù)了。同時(shí)調(diào)用了該函數(shù),系統(tǒng)會(huì)修改shmid_ds數(shù)據(jù)。

            int shmdt(const void *shmaddr)
            解除綁定關(guān)系,參數(shù)就是我們之前獲取的那個(gè)返回地址。(其實(shí)我覺(jué)得這里參數(shù)如果為ID貌似更統(tǒng)一些吧)

            int shmctl(int shmid, int cmd, struct shmid_ds *buf)
            這個(gè)函數(shù)主要做一些修改和查詢(xún)。比如設(shè)置鎖,解鎖,移除一塊共享內(nèi)存。

            簡(jiǎn)單介紹了API,還要說(shuō)一下一些注意的東西
            1.共享內(nèi)存不會(huì)把數(shù)據(jù)寫(xiě)入磁盤(pán)文件中,這個(gè)區(qū)別于mmap
            2.即使沒(méi)有進(jìn)程綁定在共享內(nèi)存,共享內(nèi)存也不會(huì)消失的。必須通過(guò)shmctl或者ipcrm刪除(或者更暴力的方式關(guān)掉電腦)

            另外我們可能會(huì)考慮,系統(tǒng)最多創(chuàng)建多少個(gè)共享內(nèi)存,一個(gè)進(jìn)程最多可以綁定多少個(gè)內(nèi)存,一個(gè)共享內(nèi)存創(chuàng)建的size最大最小值是多少。其實(shí)這些設(shè)置在/proc/sys/kernel下面,我們也可以自己寫(xiě)程序來(lái)讀取。貼一段代碼用來(lái)獲取上面的信息
            #define MAX_SHMIDS        8196

            int main(int argc,char *argv[])
            {
                int i,j;
                int shmid[MAX_SHMIDS] = {0};
                void *addr[MAX_SHMIDS] = {0};
                
                //測(cè)試可以創(chuàng)建多少個(gè)共享內(nèi)存
                for ( i = 0;i < MAX_SHMIDS;++i )
                {
                    shmid[i] = shmget(IPC_PRIVATE,1024,0666|IPC_CREAT);
                    if ( shmid[i] == -1 )
                    {
                        printf("create shared memory failed,max create num[%d],%s\r\n",i,strerror(errno));
                        break;
                    }
                }
                
                for ( int j = 0;j < i;++j )
                {
                    shmctl(shmid[j],IPC_RMID,NULL);
                }
                
                //測(cè)試每個(gè)進(jìn)程可以attach的最大數(shù)
                for ( i = 0;i < MAX_SHMIDS;++i )
                {
                    shmid[i] = shmget(IPC_PRIVATE,1024,0666|IPC_CREAT);
                    if ( shmid[i] != -1 )
                    {
                        addr[i] = shmat(shmid[i],0,0);
                        if ( addr[i] == (void *)-1 )
                        {
                            printf("process attach shared memory failed,max num[%d],%s\r\n",i,strerror(errno));
                            shmctl(shmid[i],IPC_RMID,NULL);
                            break;
                        }
                    }
                    else
                    {
                        printf("max num of process attach shared memory is[%d]\r\n",i-1);
                        break;
                    }
                }
                
                for ( j = 0;j < i;++j )
                {
                    shmdt(addr[j]);
                    shmctl(shmid[j],IPC_RMID,NULL);
                }
                
                //測(cè)試一個(gè)共享內(nèi)存創(chuàng)建最小的size
                size_t size = 0;
                for ( ;;size++ )
                {
                    shmid[0] = shmget(IPC_PRIVATE,size,0666|IPC_CREAT);
                    if ( shmid[0] != -1 )
                    {
                        printf("create shared memory succeed,min size[%d]\r\n",size);
                        shmctl(shmid[0],IPC_RMID,NULL);
                        break;
                    }
                }
                
                //測(cè)試共享內(nèi)存創(chuàng)建最大的size
                for ( size = 65536;;size += 1024 )
                {
                    shmid[0] = shmget(IPC_PRIVATE,size,0666|IPC_CREAT);
                    if ( shmid[0] == -1 )
                    {
                        printf("create shared memory failed,max size[%ld],%s\r\n",size,strerror(errno));
                        break;
                    }
                    
                    shmctl(shmid[0],IPC_RMID,NULL);
                }
                
                exit(0);
            }
            好了,下篇開(kāi)始介紹如何控制讀寫(xiě)。
            posted on 2012-09-06 19:26 梨樹(shù)陽(yáng)光 閱讀(2251) 評(píng)論(0)  編輯 收藏 引用 所屬分類(lèi): Linux
            精品无码久久久久久国产| 久久A级毛片免费观看| 国产精品欧美久久久天天影视| 国产成人久久精品激情| 久久最近最新中文字幕大全| 久久国产视屏| 亚洲国产精品无码久久一线| 欧美久久精品一级c片片| 久久影院久久香蕉国产线看观看| 亚洲∧v久久久无码精品| 国产亚州精品女人久久久久久 | 久久精品国产免费| 四虎影视久久久免费| 久久亚洲国产精品一区二区| 久久精品国产亚洲AV久| 久久久久国产日韩精品网站| 色诱久久久久综合网ywww| 久久这里只有精品视频99| 久久精品免费一区二区三区| 伊人久久大香线焦AV综合影院 | 成人a毛片久久免费播放| 无码久久精品国产亚洲Av影片| 久久伊人中文无码| 国产无套内射久久久国产| 91精品国产综合久久精品| 日日噜噜夜夜狠狠久久丁香五月| 午夜肉伦伦影院久久精品免费看国产一区二区三区 | 久久se精品一区精品二区国产| 日韩精品久久久久久久电影蜜臀| 亚洲精品99久久久久中文字幕 | 亚洲国产天堂久久综合| 久久久久久毛片免费看| 久久成人18免费网站| 久久精品国产一区二区三区不卡| 久久国产成人精品麻豆| 久久久久久久综合日本亚洲| 国产精品久久久久9999| 日本免费久久久久久久网站| 色综合久久最新中文字幕| 久久国产精品波多野结衣AV| 大蕉久久伊人中文字幕|