上節(jié)和上上節(jié)我們分享了Linux進程間通信的管道、消息隊列、信號以及信號量的基本原理和實踐,文章如下:
Linux進程間通信(上)之管道、消息隊列實踐
Linux進程間通信(中)之信號、信號量實踐
這節(jié)我們就來分享一下Linux的最后一種進程間通信的方式:共享內(nèi)存。
1、什么是共享內(nèi)存
共享內(nèi)存就是兩個不相關的進程之間可以直接訪問同一段內(nèi)存,共享內(nèi)存在兩個正在運行的進程之間共享和傳遞數(shù)據(jù)起到了非常有效的方式。在不同的進程之間共享的內(nèi)存通常安排為同一段物理內(nèi)存,進程可以將同一段共享內(nèi)存連接到它們自己的地址空間中,所有進程都可以直接訪問共享內(nèi)存中的地址。而如果某個進程向共享內(nèi)存寫入數(shù)據(jù),所做的改動將立即影響到可以訪問同一段共享內(nèi)存的任何其他進程;其實就是映射一段能夠被其它內(nèi)存所訪問到的內(nèi)存,這段內(nèi)存由一個進程創(chuàng)建,但是多個進程都可以去訪問。共享內(nèi)存是最快的IPC方式,它是通過其它通信方式的效率不足而專門設計的。往往都是和其它通信機制配合使用,來實現(xiàn)進程間的同步和通信。
共享內(nèi)存的使用和信號量其實也是差不多的,都是使用接口的形式,共享內(nèi)存的接口比信號量的接口更加的簡單,我們一起去了解下共享內(nèi)存的使用。
共享內(nèi)存函數(shù)由shmget、shmat、shmdt、shmctl
四個函數(shù)組成。我們下面來分析每一個函數(shù)的用法。
1.1、創(chuàng)建共享內(nèi)存
int shmget(key_t key, size_t size, int shmflg);
第一個參數(shù)是共享內(nèi)存段的命名,shmget成功時返回一個關于key相關的標識符,用于后續(xù)的共享內(nèi)存函數(shù)。當調(diào)用失敗返回-1。其它進程也可以通過shmget函數(shù)返回值訪問同一個共享內(nèi)存。第二個參數(shù)是指定共享內(nèi)存的容量;第三個shmflg是一個權(quán)限標志,它的作用和open和mode函數(shù)都是相同的,當共享內(nèi)存不存在的時候則通過IPC_CREAT來創(chuàng)建。共享內(nèi)存的權(quán)限標準和文件讀寫的權(quán)限一樣。
1.2、啟動對共享內(nèi)存的訪問
void *shmat(int shm_id, const void *shm_addr, int shmflg);
當我們第一次創(chuàng)建完共享內(nèi)存時,它還不能被任何進程訪問,shmat函數(shù)就是用來啟動對共享內(nèi)存的訪問,并把共享內(nèi)存連接到當前進程的地址空間。
shm_id是由shmget函數(shù)返回的共享內(nèi)存標識;shm_addr指定共享內(nèi)存連接到當前進程中的地址位置,通常為空,表示讓系統(tǒng)來選擇共享內(nèi)存的地址。最后一個參數(shù)是標志位通常都是0。調(diào)用成功時返回一個指向共享內(nèi)存第一個字節(jié)的指針,如果調(diào)用失敗返回-1。
1.3、共享內(nèi)存從當前內(nèi)存中分離
int shmdt(const void *shmaddr);
這個函數(shù)只是從共享內(nèi)存中分離而不是刪除,這一點要分清楚,對于初學者而言這里很容易掉坑,使共享內(nèi)存在當前進程中不可再用。
參數(shù)shmaddr是shmat函數(shù)返回的地址指針,調(diào)用成功時返回0,失敗時返回-1。
1.4、控制共享內(nèi)存
int shmctl(int shm_id, int command, struct shmid_ds *buf);
第一個參數(shù)是shaget函數(shù)返回的共享內(nèi)存標識符;command參數(shù)是要采取的操作,它由 IPC_STAT、IPC_SET和IPC_RMID組成,分別IPC_STAT代表把shmid_ds結(jié)構(gòu)中的數(shù)據(jù)設置為共享內(nèi)存的當前關聯(lián)值,即用共享內(nèi)存的當前關聯(lián)值覆蓋shmid_ds的值;IPC_SET代表如果進程有足夠的權(quán)限,就可以把共享內(nèi)存的當前關聯(lián)值設置為shmid_ds結(jié)構(gòu)中給出的值;IPC_RMID代表刪除共享內(nèi)存段。第三個參數(shù)buf代表一個結(jié)構(gòu)指針,它指向共享內(nèi)存的模式或訪問權(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 ;
}
//對共享內(nèi)存的訪問
shmptr = shmat(shmid, 0, 0);
if (shmptr == (void *)-1)
{
perror("shmat");
return -2 ;
}
// 往共享內(nèi)存寫數(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 ;
}
運行結(jié)果:
先分別編譯shm_snd.c和shmrcv.c這兩個程序,生成shmrcv和shmsnd這兩個可執(zhí)行程序。
接下來,首先執(zhí)行shmsnd,會得到以下結(jié)果:
什么都沒有?共享內(nèi)存創(chuàng)建成功了嗎?當然是成功了,可以通過ipcs –m
命令查看:
如圖上圖所示,nattch項下的數(shù)字為0那個就是剛剛使用shmsnd這個可執(zhí)行程序創(chuàng)建的一段共享內(nèi)存。當然,我們還往共享內(nèi)存發(fā)了shmat write ok
這個字符串,下面運行shmrcv這個程序,看看是否能把寫進共享內(nèi)存的數(shù)據(jù)讀出來。
成功讀出。同樣的,也可以刪除共享內(nèi)存,如何刪除?也一樣有兩種方法。
(1)使用ipcrm –m shmid可以刪除共享內(nèi)存
如上圖,我們已經(jīng)知道0x66的shmid為1835021,所以只要執(zhí)行ipcrm –m 1835021
命令即可刪除,如下圖所示,成功刪除。
(2)使用shmctl 函數(shù)寫入IPC_RMID指令刪除共享內(nèi)存
shmrm.c
#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
int main(void)
{
int shmid ;
//同樣,首先先打開共享內(nèi)存
shmid = shmget(0x66 , 0 , 0);
if(-1 == shmid)
{
perror("open shmkey 0x66 fail");
return -1 ;
}
//成功的話,向shmctl寫入?yún)?shù),IPC_RMID表示立刻刪除,后面的參數(shù)被忽略,為0
int ret ;
//寫入的是參數(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 ;
}
運行結(jié)果:
往期精彩
Linux進程間通信(中)之信號、信號量實踐
【Linux系統(tǒng)編程】IO標準緩沖區(qū)
替代傳統(tǒng)串口屏的Yoxios了解一下!
覺得本次分享的文章對您有幫助,隨手點[在看]
并轉(zhuǎn)發(fā)分享,也是對我的支持。
免責聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!