共享内存

共享内存是 Unix下的多进程之间的通信方法,这种方法通常用于一个程序的多进程间通信,实际上多个程序间也可以通过共享内存来传递信息。

特点

  • 所谓共享内存就是使得多个进程可以访问同一块内存空间,是最快的可用IPC形式。
  • 是针对其他通信机制运行效率较低而设计的。
  • 但内部没有共享内存互斥访问机制,所以往往与其它通信机制,如信号量结合使用,来达到进程间的同步及互斥。

声明

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
/*
fname:指定的文件名(已经存在的文件名),一般使用当前目录,如:"."
id:id是子序号。虽然是int类型,但是只使用8bits(1-255)。
在一般的UNIX实现中,是将文件的索引节点号取出,前面加上子序号得到key_t的返回值。

如:指定文件的索引节点号为65538,换算成16进制为0x010002,而你指定的ID值为38,
换算成16进制为0x26,则最后的key_t返回值为0x26010002。

查询文件索引节点号的方法是: ls -i

当删除重建文件后,索引节点号由操作系统根据当时文件系统的使用情况分配,
因此与原来不同,所以得到的索引节点号也不同。
*/

key_t ftok( const char * fname, int id );


/*
-key
	0(IPC_PRIVATE):会建立新共享内存对象
	大于0的32位整数:视参数shmflg来确定操作。通常要求此值来源于ftok返回的IPC键值

-size
	大于0的整数:新建的共享内存大小,以字节为单位
	0:只获取共享内存时指定为0

-shmflg
	0:取共享内存标识符,若不存在则函数会报错
	IPC_CREAT:当shmflg&IPC_CREAT为真时,如果内核中不存在键值与key相等的共享内存,则新建一个共享内存;如果存在这样的共享内存,返回此共享内存的标识符
	IPC_CREAT|IPC_EXCL:如果内核中不存在键值与key相等的共享内存,则新建一个共享内存;如果存在这样的共享内存则报错
使用时需要与IPC对象存取权限(如0600)进行|运算来确定信号量集的存取权限

-函数返回值
	成功:返回共享内存的标识符
	出错:-1,错误原因存于error中

-错误代码
	EINVAL:参数size小于SHMMIN或大于SHMMAX
	EEXIST:预建立key所指的共享内存,但已经存在
	EIDRM:参数key所指的共享内存已经删除
	ENOSPC:超过了系统允许建立的共享内存的最大值(SHMALL)
	ENOENT:参数key所指的共享内存不存在,而参数shmflg未设IPC_CREAT位
	EACCES:没有权限
	ENOMEM:核心内存不足
*/
int shmget(key_t key, size_t size, int shmflg);

/*
shmid:共享内存标识符

cmd
IPC_STAT:得到共享内存的状态,把共享内存的shmid_ds结构复制到buf中
IPC_SET:改变共享内存的状态,把buf所指的shmid_ds结构中的uid、gid、mode复制到共享内存的shmid_ds结构内
IPC_RMID:删除这片共享内存

buf:共享内存管理结构体。具体说明参见共享内存内核结构定义部分

函数返回值
成功:0
出错:-1,错误原因存于error中

错误代码
EACCESS:参数cmd为IPC_STAT,确无权限读取该共享内存
EFAULT:参数buf指向无效的内存地址
EIDRM:标识符为shmid的共享内存已被删除
EINVAL:无效的参数cmd或shmid
EPERM:参数cmd为IPC_SET或IPC_RMID,却无足够的权限执行
*/
int shmctl(int shmid, int cmd, struct shmid_ds *buf)

例子:

写端:

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>

int main(int argc, const char * argv[]) {
    
    //生成一个key
    key_t key = ftok("./", 88);
    
    //创建共享内存,返回一个id
    //数字 4 、2 和 1表示读、写、执行权限
    //用户、所属组、其他组都有读写权限
    int shmid = shmget(key, 8, IPC_CREAT|0666);
    // IPC_CREAT: Create entry if key does not exist
    
    if (shmid == -1) {
        perror("shmget failed");
        //exit(0) 表示程序正常退出,exit⑴/exit(-1)表示程序异常退出。
        exit(1);
    }
    
    //映射共享内存,得到虚拟地址
    //shmaddr:指定共享内存出现在进程内存地址的什么位置,直接指定为NULL让内核自己决定一个合适的地址位置
    //shmflg: SHM_RDONLY:为只读模式,其他为读写模式
    void *p = shmat(shmid, NULL, 0);
    if (p == (void *)-1) {
        perror("shmat failed");
        exit(2);
    }
    
    //写共享内存
    int *pp = p;
    *pp = 0x123456;
    *(pp + 1) = 0xffffff;
    
    //解除映射
    if (shmdt(p) == -1) {
        printf("shmdt failed");
        exit(3);
    }
    
    printf("解除映射成功,点击回车销毁共享内存\n");
    getchar();
    
    //IPC_RMID:删除这片共享内存
    if (shmctl(shmid, IPC_RMID, NULL)) {
        perror("shmctl failed");
        exit(4);
    }
    
    return 0;
}

读端:

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdlib.h>

int main() {
    
    //生成key
    key_t key = ftok("./", 88);
    
    //获取共享内存,返回id
    //0:只获取共享内存时指定为0
    //shmflg0:取共享内存标识符,若不存在则函数会报错
    int shmid = shmget(key, 0, 0);
    if (shmid == -1) {
        perror("shmget failed");
        exit(1);
    }
    
    //映射共享内存,得到虚拟地址
    void *p = shmat(shmid, 0, 0);
    if (p == (void *)-1) {
        perror("shmat failed");
        exit(2);
    }
    
    //读取共享内存
    int data1 = *(int *)p;
    int data2 = *((int *)p + 1);
    
    printf("从共享内存中读取了,%x 和 %x\n", data1, data2);
    
    //解除映射
    if(shmdt(p) == -1) {
        perror("shmdt failed");
        exit(3);
    }
    
    
    return 0;
}