System V共享内存介绍
(一)简单概念
共享内存作为一种进程间通信的方式,其相较于其他进程间通信方式而言最大的优点就是数据传输速率快。其内部实现的方式采用了Linux进程地址空间中的mmap文件映射区,将文件内容直接映射到各自进程的进程地址空间中,进程对各自进程地址空间的访问即可
完成数据通信,由于直接读取内存的方式,故其效率远远快于其他IPC方法,这是共享内存的一大优势。但对于共享内存来说,保证数据同步问题是一个难点,在一般情况下可以采用管道的方式,本节内容的实例代码采用了互斥锁机制。
(二)System V共享内存API函数
(1)int shmget(key_t key, ssize_t size, int oflag)
功能:用于创建共享内存区域。
参数:
key:共享内存的名字,用于唯一确定一个共享内存;
size:创建的共享内存的大小;
oflag:共享内存的读写权限值集合,与文件创建中的mode标志是一样的。同样还可以与IPC_CREAT,IPC_EXCL等标志联合使用。
返回:函数返回共享内存区的标识符。
(2)void *shmat(int shmid, const void *shmaddr, int flag)
功能:将一个创建好的共享内存区附接到进程对应的进程地址空间中。
参数:
shmid:shmget函数的返回值;
shmaddr:将共享内存附接道shmaddr指定的进程地址空间的对应地址上。如果设置为NULL,将由系统指定合法的区域将共享内存映射,这是推荐的做法;
flag:两个可能取值为SHM_RND和SHM_RDONLY。
返回:函数返回映射区的起始地址。
(3)int shmdt(const void *shmaddr)
功能:将共享内存从该进程地址空间中脱离。
参数:
shmaddr:shmat函数的返回值。
返回:如果成功,函数返回0,出错返回-1。
注意:该函数只是将共享内存从进程的地址空间中脱离,不会删除该共享内存区域。并且当一个进程终止时,它当前附接的所有共享内存区都将会自动脱离。
(4)int shmctl(int shmid, int cmd, struct shmid_ds *buf)
功能:提供对共享内存区的多种操作
参数:
shmid:shmget函数的返回值;
cmd:具体操作。取值有1)IPC_RMID:从系统中删除shmid标识的共享内存区并拆除;2)IPC_SET:在进程有足够权限的前提下,把共享内存的当前关联值设置为shmid_ds数据结构中给出的值;3)IPC_START:向调用者返回指定的共享内存区当前的shmid_ds结构。
返回:成功返回0,出错返回-1。
(三)采用互斥锁与共享内存的进程间通信方式
由于共享内存需要解决的一个问题就是数据同步,进程间的同步方式采用信号量可以完成,但我考虑能否采用互斥锁的方式。这里采用互斥锁最初的疑惑为:两个没有近亲关系的进程,如何能够获取到同一个互斥锁(一般互斥锁用于线程间同步的时候比较多,这里也说回来,线程之间本来就是共享数据,所以只需要解决数据同步问题即可,故还没有听说过用共享内存来解决线程通信的 - -。)。
这里来看下,如果设置互斥锁pthread_mutex_t,来使得不同进程可以获取到同一个互斥锁。
对于一个互斥锁pthread_mutex_t来说,有两种方式进行初始化,一种是静态分配,一种是动态初始化。
1)利用宏PTHREAD_MUTEX_INITALIZER来初始化静态分配的互斥锁
pthread_mutex_t mutex = PTHREAD_MUTEX_INITALIZER
2)利用pthread_mutex_init来动态初始化互斥锁
函数原型:int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *mutexattr)
这里第一个参数mutex很好理解,就是定义的需要初始化的互斥锁。这里的第二个参数mutexattr代表什么?Linux下互斥锁具有一系列的属性,其中pthread_mutexattr_t结构体定义了一套完整的互斥锁属性。两种常用的属性是pshared和type。其中pshared属性指定了是否允许跨进程共享互斥锁,可选值为:1)PTHREAD_PROCESS_SHARED,互斥锁可以被跨进程共享;2)PTHREAD_PROCESS_PRIVATE,互斥锁只能隶属于一个进程,默认属性。
我们可以通过设置互斥锁自身属性,来达到跨进程共享互斥锁的方法。
结合实例代码来进行分析:
//shmat.h #ifndef __SHMAT_H__
#define __SHMAT_H__ #include <pthread.h>
#include <string.h>
#include <sys/shm.h>
#include <stdlib.h> #define SM_BUF_SIZE 1024
#define SM_ID 0x1234 struct shmat_msg
{
int flag;
pthread_mutex_t shmat_mutex;
char buf[SM_BUF_SIZE];
}; #endif
头文件声明了共享内存结构体,其中shmat_mutex用于保证对共享内存的互斥访问。
//shmat-read.c #include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h> #include "shmat.h" int main(void)
{
int running = ;
int shm_id, ret;
void *shared_memory = NULL;
struct shmat_msg *sm_msg = NULL; pthread_mutexattr_t attr;
pthread_mutexattr_init(&attr);
pthread_mutexattr_setpshared(&attr, PTHREAD_PROCESS_SHARED); //设置跨进程属性 shm_id = shmget((key_t)SM_ID, sizeof(struct shmat_msg), | IPC_CREAT); //创建共享内存,返回共享内存标识符
if (shm_id < )
{
perror("fail to shmget\n");
exit();
} shared_memory = shmat(shm_id, NULL, ); //映射到进程虚拟地址空间
if (shared_memory == NULL)
{
perror("fail to shmat\n");
exit();
}
sm_msg = (struct shmat_msg*)shared_memory; //转型为shmat_msg数据结构
sm_msg->flag = ;
pthread_mutex_init(&sm_msg->shmat_mutex, &attr); //初始化共享内存的互斥锁 while (running)
{
pthread_mutex_lock(&sm_msg->shmat_mutex); //互斥访问
if (sm_msg->flag)
{
printf("read message : %s\n", sm_msg->buf);
sm_msg->flag = ;
if (strncmp(sm_msg->buf, "exit", ) == )
running = ;
pthread_mutex_unlock(&sm_msg->shmat_mutex);
}
else
{
printf("no data to read, waiting\n");
pthread_mutex_unlock(&sm_msg->shmat_mutex);
sleep();
}
} ret = shmdt(shared_memory); //脱离共享内存的映射
if (ret < )
{
perror("failed to shmdt\n");
exit();
} if (shmctl(shm_id, IPC_RMID, ) < ) //删除共享内存
{
perror("failed to shmctl\n");
exit();
} return ;
}
读文件。设置了共享内存互斥锁为跨进程属性。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h> #include "shmat.h" int main(void)
{
int running = ;
int shm_id, ret;
void *shared_memory = NULL;
struct shmat_msg *sm_msg = NULL; shm_id = shmget((key_t)SM_ID, sizeof(struct shmat_msg), ); //获取共享内存
if (shm_id < )
{
perror("failed to shmget\n");
exit();
} shared_memory = shmat(shm_id, NULL, ); //映射
if (shared_memory == NULL)
{
perror("failed to shmat\n");
exit();
} sm_msg = (struct shmat_msg*)shared_memory;
char buff[]; while (running)
{
pthread_mutex_lock(&sm_msg->shmat_mutex); //获取共享内存的互斥锁
if (sm_msg->flag == )
{
fgets(buff, , stdin);
printf("write sm_msg : %s\n", buff);
strncpy(sm_msg->buf, buff, sizeof(buff));
sm_msg->flag = ;
if (strncmp(sm_msg->buf, "exit", ) == )
running = ;
pthread_mutex_unlock(&sm_msg->shmat_mutex);
}
else
{
printf("sm_msg waiting read\n");
pthread_mutex_unlock(&sm_msg->shmat_mutex);
sleep();
}
} ret = shmdt(shared_memory);
if (ret < )
{
perror("failed to shmdt\n");
exit();
} ret = shmctl(shm_id, IPC_RMID, );
if (ret < )
{
perror("failed to shmctl\n");
exit();
} return ;
}
写文件。
程序运行结果:
其中shmat-write程序运行后会阻塞在fgets等待用户输入。其中shmat-read与shmat-write两个程序抢占lock的时机是不确定的,可能同一个程序会多次抢占互斥锁,但由于没有内容可读或可写,则睡眠等待。
unlock解锁后,操作系统下次调度哪个进程加锁是不一定的,如果需要在满足一定条件后才被调度可以采用条件变量(但一般都是在多线程环境下了)。
System V共享内存介绍的更多相关文章
- Linux进程通信之System V共享内存
前面已经介绍过了POSIX共享内存区,System V共享内存区在概念上类似POSIX共享内存区,POSIX共享内存区的使用是调用shm_open创建共享内存区后调用mmap进行内存区的映射,而Sys ...
- 阐述linux IPC(五岁以下儿童):system V共享内存
[版权声明:尊重原创.转载请保留源:blog.csdn.net/shallnet 要么 .../gentleliu,文章学习交流,不用于商业用途] system V共享内存和posix ...
- UNIX环境高级编程——System V 共享内存区
共享内存区域是被多个进程共享的一部分物理内存.如果多个进程都把该内存区域映射到自己的虚拟地址空间,则这些进程就都可以直接访问该共享内存区域,从而可以通过该区域进行通信.共享内存是进程间共享数据的一种最 ...
- Linux IPC实践(9) --System V共享内存
共享内存API #include <sys/ipc.h> #include <sys/shm.h> int shmget(key_t key, size_t size, int ...
- php进程(线程)通信基础--System V共享内存
PHP默认情况没有开启功能,要支持该功能在编译PHP的时候要加入下面几个选项 System V消息,--enable-sysvmsg System V信号量支持,--enable-sysvsem ...
- System V 共享内存区
1.概述 系统调用mmap通过映射一个普通文件实现共享内存.System V 则是通过映射特殊文件系统shm中的文件实现进程间的共享内存通信.也就是说,每个共享内存区域对应特殊文件系统shm中的一个文 ...
- 共享内存之——system V共享内存
System V 的IPC对象有共享内存.消息队列.信号灯(量). 注意:在IPC的通信模式下,不管是共享内存.消息队列还是信号灯,每个IPC的对象都有唯一的名字,称为"键(key)&quo ...
- Linux system v 共享内存
system v 共享内存 #include <sys/types.h> #include <sys/shm.h> int shmget(key_t key, size_t s ...
- System V共享内存
目录 1. 概述 2. System V共享内存API shmget shmat shmdt shmctl 3. 简单的程序 代码实现 common.h shmcreate.c shmrmid.c s ...
随机推荐
- IO Model- 同步,异步,阻塞,非阻塞
同步(synchronous) IO和异步(asynchronous) IO,阻塞(blocking) IO和非阻塞(non-blocking)IO分别是什么,到底有什么区别?这个问题其实不同的人给出 ...
- bzoj4501 旅行
题面: 小C来到了F国,小C想好好地参观F国.F国可以看一个有n个点m条边的有向无环图,小C刚开始站在1号点.假设现在小C站在x号点: 1.点x没有出边,结束旅游. 2.点x有o条出边,小C等概率地选 ...
- javascript中面向对象的5种写法
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- BZOJ4765 普通计算姬(分块+树状数组)
对节点按编号分块.设f[i][j]为修改j号点对第i块的影响,计算f[i][]时dfs一遍即可.记录每一整块的sum.修改时对每一块直接更新sum,同时用dfs序上的树状数组维护子树和.查询时累加整块 ...
- javascript中var let const三种变量声明方式
javascript中var let const三种变量声明方式 1.var ①var表示声明了一个变量,并且可以同时初始化该变量. ②使用var语句声明的变量的作用域是当前执行位置的上下文:一个函 ...
- 【QQ】前端实现QQ会话功能
<a href="tencent://message/?uin=客服QQ号码&Menu=yes" target="blank"></a ...
- BZOJ1877:[SDOI2009]晨跑——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=1877 https://www.luogu.org/problemnew/show/P2153 Ela ...
- [zhuan]tomcat环境配置
http://jingyan.baidu.com/article/8065f87fcc0f182330249841.html 一.安装JDK和Tomcat 1,安装JDK:直接运行jdk-7-wind ...
- JavaScript去除数组中的重复值
用原型函数(prototype)可以定义一些很方便的自定义函数,实现各种自定义功能. Javascript 中的原型函数(prototype)的工作原理,在 javascript 中每次声明新函数的过 ...
- 数据添加到solr索引库后前台如何搜索
主要结构: 查询 Dao: package com.taotao.search.dao.impl; import java.util.ArrayList; import java.util.List; ...