當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > 嵌入式云IOT技術(shù)圈
[導(dǎo)讀]上節(jié)和上上節(jié)我們分享了Linux進(jìn)程間通信的管道、消息隊(duì)列、信號(hào)以及信號(hào)量的基本原理和實(shí)踐,文章如下: Linux進(jìn)程間通信(上)之管道、消息隊(duì)列實(shí)踐 Linux進(jìn)程間通信(中)之信號(hào)、信號(hào)量實(shí)踐 這節(jié)我們就來(lái)分享一下Linux的最后一種進(jìn)程間通信的方式:共享內(nèi)存。


上節(jié)和上上節(jié)我們分享了Linux進(jìn)程通信的管道、消息隊(duì)列、信號(hào)以及信號(hào)量的基本原理和實(shí)踐,文章如下:

Linux進(jìn)程間通信(上)之管道、消息隊(duì)列實(shí)踐

Linux進(jìn)程間通信(中)之信號(hào)、信號(hào)量實(shí)踐

這節(jié)我們就來(lái)分享一下Linux的最后一種進(jìn)程間通信的方式:共享內(nèi)存。

1、什么是共享內(nèi)存

共享內(nèi)存就是兩個(gè)不相關(guān)的進(jìn)程之間可以直接訪問(wèn)同一段內(nèi)存,共享內(nèi)存在兩個(gè)正在運(yùn)行的進(jìn)程之間共享和傳遞數(shù)據(jù)起到了非常有效的方式。在不同的進(jìn)程之間共享的內(nèi)存通常安排為同一段物理內(nèi)存,進(jìn)程可以將同一段共享內(nèi)存連接到它們自己的地址空間中,所有進(jìn)程都可以直接訪問(wèn)共享內(nèi)存中的地址。而如果某個(gè)進(jìn)程向共享內(nèi)存寫(xiě)入數(shù)據(jù),所做的改動(dòng)將立即影響到可以訪問(wèn)同一段共享內(nèi)存的任何其他進(jìn)程;其實(shí)就是映射一段能夠被其它內(nèi)存所訪問(wèn)到的內(nèi)存,這段內(nèi)存由一個(gè)進(jìn)程創(chuàng)建,但是多個(gè)進(jìn)程都可以去訪問(wèn)。共享內(nèi)存是最快的IPC方式,它是通過(guò)其它通信方式的效率不足而專門設(shè)計(jì)的。往往都是和其它通信機(jī)制配合使用,來(lái)實(shí)現(xiàn)進(jìn)程間的同步和通信。

共享內(nèi)存的使用和信號(hào)量其實(shí)也是差不多的,都是使用接口的形式,共享內(nèi)存的接口比信號(hào)量的接口更加的簡(jiǎn)單,我們一起去了解下共享內(nèi)存的使用。

共享內(nèi)存函數(shù)由shmget、shmat、shmdt、shmctl四個(gè)函數(shù)組成。我們下面來(lái)分析每一個(gè)函數(shù)的用法。

1.1、創(chuàng)建共享內(nèi)存

int shmget(key_t key, size_t size, int shmflg);

第一個(gè)參數(shù)是共享內(nèi)存段的命名,shmget成功時(shí)返回一個(gè)關(guān)于key相關(guān)的標(biāo)識(shí)符,用于后續(xù)的共享內(nèi)存函數(shù)。當(dāng)調(diào)用失敗返回-1。其它進(jìn)程也可以通過(guò)shmget函數(shù)返回值訪問(wèn)同一個(gè)共享內(nèi)存。第二個(gè)參數(shù)是指定共享內(nèi)存的容量;第三個(gè)shmflg是一個(gè)權(quán)限標(biāo)志,它的作用和open和mode函數(shù)都是相同的,當(dāng)共享內(nèi)存不存在的時(shí)候則通過(guò)IPC_CREAT來(lái)創(chuàng)建。共享內(nèi)存的權(quán)限標(biāo)準(zhǔn)和文件讀寫(xiě)的權(quán)限一樣。

1.2、啟動(dòng)對(duì)共享內(nèi)存的訪問(wèn)

void *shmat(int shm_id, const void *shm_addr, int shmflg);

當(dāng)我們第一次創(chuàng)建完共享內(nèi)存時(shí),它還不能被任何進(jìn)程訪問(wèn),shmat函數(shù)就是用來(lái)啟動(dòng)對(duì)共享內(nèi)存的訪問(wèn),并把共享內(nèi)存連接到當(dāng)前進(jìn)程的地址空間。

shm_id是由shmget函數(shù)返回的共享內(nèi)存標(biāo)識(shí);shm_addr指定共享內(nèi)存連接到當(dāng)前進(jìn)程中的地址位置,通常為空,表示讓系統(tǒng)來(lái)選擇共享內(nèi)存的地址。最后一個(gè)參數(shù)是標(biāo)志位通常都是0。調(diào)用成功時(shí)返回一個(gè)指向共享內(nèi)存第一個(gè)字節(jié)的指針,如果調(diào)用失敗返回-1。

1.3、共享內(nèi)存從當(dāng)前內(nèi)存中分離

int shmdt(const void *shmaddr);

這個(gè)函數(shù)只是從共享內(nèi)存中分離而不是刪除,這一點(diǎn)要分清楚,對(duì)于初學(xué)者而言這里很容易掉坑,使共享內(nèi)存在當(dāng)前進(jìn)程中不可再用。

參數(shù)shmaddr是shmat函數(shù)返回的地址指針,調(diào)用成功時(shí)返回0,失敗時(shí)返回-1。

1.4、控制共享內(nèi)存

int shmctl(int shm_id, int command, struct shmid_ds *buf);

第一個(gè)參數(shù)是shaget函數(shù)返回的共享內(nèi)存標(biāo)識(shí)符;command參數(shù)是要采取的操作,它由 IPC_STAT、IPC_SET和IPC_RMID組成,分別IPC_STAT代表把shmid_ds結(jié)構(gòu)中的數(shù)據(jù)設(shè)置為共享內(nèi)存的當(dāng)前關(guān)聯(lián)值,即用共享內(nèi)存的當(dāng)前關(guān)聯(lián)值覆蓋shmid_ds的值;IPC_SET代表如果進(jìn)程有足夠的權(quán)限,就可以把共享內(nèi)存的當(dāng)前關(guān)聯(lián)值設(shè)置為shmid_ds結(jié)構(gòu)中給出的值;IPC_RMID代表刪除共享內(nèi)存段。第三個(gè)參數(shù)buf代表一個(gè)結(jié)構(gòu)指針,它指向共享內(nèi)存的模式或訪問(wèn)權(quán)限的結(jié)構(gòu)。

shmid_ds結(jié)構(gòu)至少包括以下成員:

struct shmid_ds  
{  
  uid_t shm_perm.uid;
  uid_t shm_perm.gid;
  mode_t shm_perm.mode;
};

2、共享內(nèi)存案例

shm_snd.c

#include <stdlib.h>
#include <string.h>
#include <signal.h>
#include <fcntl.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#define SHM_SIZE 1024

int main()
{
        int shmid;
        char *shmptr;   
        //創(chuàng)建共享內(nèi)存
        shmid = shmget(0x66, SHM_SIZE, IPC_CREAT|0666);
        //創(chuàng)建失敗
        if(shmid < 0)
        {               
            perror("shmget");
            return -1 ;
        }
        //對(duì)共享內(nèi)存的訪問(wèn)
        shmptr = shmat(shmid, 0, 0);
         if (shmptr == (void *)-1)
        {
            perror("shmat");
            return -2 ;
        }
        // 往共享內(nèi)存寫(xiě)數(shù)據(jù)
        strcpy(shmptr, "shmat write ok");
        shmdt(shmptr);
        return 0 ;
}

shm_rcv.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <signal.h>
#include <fcntl.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#define SHM_SIZE 1024

int main()
{
    int shmid;
    char *shmptr;
    shmid = shmget(0x66, SHM_SIZE, IPC_CREAT|0666);
    if(shmid < 0)
    {
        perror("shmget");
        return -1 ;
    }
    shmptr = shmat(shmid, 0, 0);
    if (shmptr == (void *)-1)
    {
        perror("shmat");
        return -2 ;
    }
    // 從共享內(nèi)存讀數(shù)據(jù)
    printf("read:%s\n", shmptr);
    shmdt(shmptr);
    return 0 ;
}

運(yùn)行結(jié)果:

先分別編譯shm_snd.c和shmrcv.c這兩個(gè)程序,生成shmrcv和shmsnd這兩個(gè)可執(zhí)行程序。

接下來(lái),首先執(zhí)行shmsnd,會(huì)得到以下結(jié)果:

什么都沒(méi)有?共享內(nèi)存創(chuàng)建成功了嗎?當(dāng)然是成功了,可以通過(guò)ipcs –m命令查看:

如圖上圖所示,nattch項(xiàng)下的數(shù)字為0那個(gè)就是剛剛使用shmsnd這個(gè)可執(zhí)行程序創(chuàng)建的一段共享內(nèi)存。當(dāng)然,我們還往共享內(nèi)存發(fā)了shmat write ok這個(gè)字符串,下面運(yùn)行shmrcv這個(gè)程序,看看是否能把寫(xiě)進(jìn)共享內(nèi)存的數(shù)據(jù)讀出來(lái)。

成功讀出。同樣的,也可以刪除共享內(nèi)存,如何刪除?也一樣有兩種方法。

(1)使用ipcrm –m shmid可以刪除共享內(nèi)存

如上圖,我們已經(jīng)知道0x66的shmid為1835021,所以只要執(zhí)行ipcrm –m 1835021命令即可刪除,如下圖所示,成功刪除。

(2)使用shmctl 函數(shù)寫(xiě)入IPC_RMID指令刪除共享內(nèi)存

shmrm.c

#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
int main(void)
{
        int shmid ;
        //同樣,首先先打開(kāi)共享內(nèi)存
        shmid = shmget(0x66 , 0 , 0);
        if(-1 == shmid)
        {
                perror("open  shmkey 0x66 fail");
                return -1 ;
        }
        //成功的話,向shmctl寫(xiě)入?yún)?shù),IPC_RMID表示立刻刪除,后面的參數(shù)被忽略,為0
        int ret ;
        //寫(xiě)入的是參數(shù)
        ret = shmctl(shmid , IPC_RMID , NULL);
        if(ret < 0)
        {
                perror("remove shm fail");
                return -2 ;
        }
        printf("remove key:%d success ... \n" , 0x66);
        return 0 ;
}

運(yùn)行結(jié)果:

往期精彩

Linux進(jìn)程間通信(中)之信號(hào)、信號(hào)量實(shí)踐

Linux進(jìn)程通信(上)之管道、消息隊(duì)列實(shí)踐

【Linux系統(tǒng)編程】IO標(biāo)準(zhǔn)緩沖區(qū)

替代傳統(tǒng)串口屏的Yoxios了解一下!

覺(jué)得本次分享的文章對(duì)您有幫助,隨手點(diǎn)[在看]并轉(zhuǎn)發(fā)分享,也是對(duì)我的支持。

免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問(wèn)題,請(qǐng)聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉