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 ...
随机推荐
- mysql通过binlog恢复数据
如果mysql不小心操作失误导致数据错误或者丢失这时候binlog起到了很大的作用 恢复有几种方式 1.按时间恢复--start-datetime 如果确定了时间点,那么按时间恢复是一个再好不过的 ...
- mysql,oracle,sql server中的默认事务隔离级别查看,更改
未提交读(隔离事务的最低级别,只能保证不读取物理上损坏的数据) 已提交读(数据库引擎的默认级别) 可重复读 可序列化(隔离事务的最高级别,事务之间完全隔离) 可串行化比较严谨,级别高; MySQL m ...
- SQL SERVER 存储过程中SELECT 返回值如何赋值给变量
今天在处理一个问题时,使用到一个存储过程,是用于更新并获取最新ID的.在使用过程中,需要获取到这个ID并赋值给变量,结果用EXEC @ID = 存储过程的方式获取失败了.具体情况如下: 为了还原整个情 ...
- 【Json】C#格式化JSON字符串
很多时候我们需要将json字符串以 { "status": 1, "sum": 9 }这种方式显示,而从服务端取回来的时候往往是这样 {&quo ...
- BZOJ 1045 糖果传递(思维)
设第i个人给了第i+1个人mi个糖果(可以为负),因为最后每个人的糖果都会变成sum/n. 可以得到方程组 mi-mi+1=a[i+1]-sum/n.(1<=i<=n). 把方程组化为mn ...
- 【bzoj1304】[CQOI2009]叶子的染色 树形dp
题目描述 给一棵m个结点的无根树,你可以选择一个度数大于1的结点作为根,然后给一些结点(根.内部结点和叶子均可)着以黑色或白色.你的着色方案应该保证根结点到每个叶子的简单路径上都至少包含一个有色结点( ...
- 【bzoj3754】Tree之最小方差树 最小生成树
题目描述 给出一张无向图,求它的一棵生成树,使得选出的所有边的方差最小.输出这个最小方差. 输入 第一行两个正整数N,M 接下来M行,每行三个正整数Ui,Vi,Ci N<=100,M<=2 ...
- 【bzoj1334】[Baltic2008]Elect 背包dp
题目描述 N个政党要组成一个联合内阁,每个党都有自己的席位数. 现在希望你找出一种方案,你选中的党的席位数要大于总数的一半,并且联合内阁的席位数越多越好. 对于一个联合内阁,如果某个政党退出后,其它党 ...
- 封装一个jquery库
现在Javascript库海量,流行的也多,比如jQuery,YUI等,虽然功能强大,但也是不万能的,功能不可能涉及方方面面,自己写一个的JS库是对这些的补充,很多也比较实用,把应用到项目中中去也比较 ...
- Give NetScaler a “Tune-Up”
Give NetScaler a “Tune-Up” https://www.citrix.com/blogs/2014/10/21/give-netscaler-a-tune-up/ To Opti ...