您好,登錄后才能下訂單哦!
共享內(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)存后需要掛接)
函數(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)存標識符 | |
cmd | IPC_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; }
免責聲明:本站發(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)容。