溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

進程間通信——共享內(nèi)存

發(fā)布時間:2020-07-07 15:14:32 來源:網(wǎng)絡(luò) 閱讀:497 作者:LHSTS 欄目:編程語言

共享內(nèi)存


進程間通信的本質(zhì)是讓不同的進程訪問一塊公共的資源。

1、共享內(nèi)存是進程間通信最快的方式(為什么)

2、共享內(nèi)存不提供任何的同步與互斥關(guān)系。(由用戶維護,可以用信號量)

以下圖解釋了問題1,原因是,由于共享內(nèi)存的機制,兩個進程不需要拷貝拷貝數(shù)據(jù),這個特點可能在數(shù)據(jù)較少的情況下看不出來,但是數(shù)據(jù)較多時,優(yōu)勢較為明顯。

下圖是shmat之前之后的共享內(nèi)存示意圖:(shmget獲得共享內(nèi)存后需要掛接)

進程間通信——共享內(nèi)存

進程間通信——共享內(nèi)存


函數(shù):

用于Linux進程通信共享內(nèi)存。共享內(nèi)存函數(shù)由shmget、shmat、shmdt、shmctl四個函數(shù)組成。

shmat(把共享內(nèi)存區(qū)對象映射到調(diào)用進程的地址空間)

void *shmat(int shmid, const void *shmaddr, int shmflg)

參數(shù):

shmid共享內(nèi)存標識符
shmaddr指定共享內(nèi)存出現(xiàn)在進程內(nèi)存地址的什么位置,直接指定為NULL讓內(nèi)核自己決定一個合適的地址位置
shmflg如果設(shè)置為SHM_RDONLY是只讀模式,其他為讀寫模式

返回值:成功返回附加好的共享內(nèi)存地址

shmdt(斷開共享內(nèi)存連接)

int shmdt(const void *shmaddr)

參數(shù)

shmaddr:連接的共享內(nèi)存的起始地址

返回值:成功返回0

shmget(得到一個共享內(nèi)存標識符或創(chuàng)建一個共享內(nèi)存對象)

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

參數(shù)

key大于0的32位整數(shù):視參數(shù)shmflg來確定操作。通常要求此值來源于ftok返回的IPC鍵值
size大于0的整數(shù):新建的共享內(nèi)存大小,以字節(jié)為單位
flags有IPC_CREAT和IPC_EXCL(用法同前面寫的一致)

返回值:成功返回共享內(nèi)存標示符

注:system V分配內(nèi)存的方法以頁為基本單位,一般以頁的整數(shù)倍分配。

shmctl完成對共享內(nèi)存的控制

int shmctl(int shmid, int cmd, struct shmid_ds *buf)

參數(shù):


shmid

共享內(nèi)存標識符

cmdIPC_STAT:得到共享內(nèi)存的狀態(tài),把共享內(nèi)存的shmid_ds結(jié)構(gòu)復制到buf中
IPC_SET:改變共享內(nèi)存的狀態(tài),把buf所指的shmid_ds結(jié)構(gòu)中的uid、gid、mode復制到共享內(nèi)存的shmid_ds結(jié)構(gòu)內(nèi)
IPC_RMID:刪除這片共享內(nèi)存
buf
共享內(nèi)存管理結(jié)構(gòu)體。具體說明參見共享內(nèi)存內(nèi)核結(jié)構(gòu)定義部分

共享內(nèi)存實現(xiàn)通信的例子

comm.h

#pragma once
#include<errno.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<string.h>
#define _PATH_NAME_ "/temp"
#define _PROJ_ID_ 0x6666

int create_shm(int size);
int get_shm();
int destory_shm(int shm_id);
void* shm_at(int shm_id);
int shm_dt(void*shmaddr);

comm.c

#include"comm.h"

static int comm_create_shm(int size,int flags)
{
    key_t _key = ftok(_PATH_NAME_,_PROJ_ID_);
    if(_key<0)
    {
        perror("ftok");
        return -1;
    }
//    int shm_id=shmget(_key,size,IPC_CREAT | IPC_EXCL);

    int shm_id=shmget(_key,size,flags);
    if(shm_id < 0)
    {
        perror("shmget");
        return -2;
    }
    return shm_id;
}

int create_shm(int size)
{
    int flags = IPC_CREAT | IPC_EXCL|0666;
    return  comm_create_shm(size,flags);
}
int get_shm()
{
    
    int flags = IPC_CREAT;
    return  comm_create_shm(0,flags);
}
int destory_shm(int shm_id)
{
    if(shmctl(shm_id,IPC_RMID,NULL)<0)
    {    
        perror("shmctl");
        return -1;
    }
    return 0;
}

void* shm_at(int shm_id)
{
    return shmat(shm_id,NULL,0);
}
int shm_dt(void* shmaddr)
{
    return shmdt(shmaddr);
}

server.c

#include"comm.c"

int main()
{
    int shm_id = create_shm(4096);
    sleep(5);
    char* buf = (char*)shm_at(shm_id);
    sleep(1);
    sleep(1);
    while(1)
    {
        printf("%s\n",buf);
        sleep(1);
        if(strcmp(buf,"AAAAA") == 0)
        {
            break;
        }
    }
    shm_dt(buf);
    sleep(5);
    destory_sem(sem_id);
    return 0;
}

client.c

#include"comm.c"

int main()
{
    int shm_id = get_shm();
    char* buf = (char*)shm_at(shm_id);
    int index = 0;
    while(1)
    {
        buf[index++]='A';
        buf[index]='\0';
        sleep(1);
        if(index>5)
        {
            break;
        }
    }
    sleep(5);
    shm_dt(buf);
    sleep(5);
    return 0;
}








向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI