System V共享内存
1. 概述
System V共享内存在概念上类似于Posix共享内存,代之以调用shm_Open后调用mmap的是,先调用shmget,再调用shmat。
对于每个System V共享内存,内核都维护如下的信息结构,它定义在sys/shm.h头文件中,其中带注释的是我们需要关注的成员。
struct shmid_ds
{
struct ipc_perm shm_perm;
size_t shm_segsz; //共享内存区大小
pid_t shm_lpid;
pid_t shm_cpid;
shmatt_t shm_nattch;
shmat_t shm_cnattch;
time_t shm_atime;
time_t shm_dtime;
time_t shm_ctime;
};
2. System V共享内存API
shmget
shmget用于创建一个新的共享内存或打开一个已存在的共享内存。
//成功返回共享内存标识符,
int shmget(key_t key, size_t size, int oflag);
- 参数size是共享内存区大小,其余两个参数含义及用法和System V信号量一样
- 当实际操作为创建新的共享内存时,该内存区size个字节均被初始化为0
- 当实际操作为打开已有共享内存时,size可设为0,oflag设为需要的读写权限
shmat
shmat用于把shmget创建或打开的共享内存连接到调用进程的地址空间。
//成功返回映射区起始地址,失败返回-1
void *shmat(int shmid, const void *shmaddr, int flag);
- shmid是shmget返回的标识符
- shmaddr推荐设为NULL,表示由系统决定映射区起始地址
- flag一般设为0,因为只要调用进程具有共享内存的读写权限,那么映射区内存就也可以读写
- flag也可以设为SHM_RDONLY限定只读访问
shmdt
shmdt删除由shmat建立的连接。
//成功返回0,失败返回-1
int shmdt(const void *shmaddr);
shmctl
shmctl用于对共享内存的各种控制操作。
//成功返回0,失败返回-1
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
cmd可使用的命令有三个:
- IPC_RMID:从系统中删除共享内存,此时buf参数设为NULL即可
- IPC_STAT:通过buf返回共享内存对应的shmid_ds结构,一般用此命令获取共享内存区大小
- IPC_SET:通过buf设置共享内存对应shmid_ds结构中的shm_perm.uid、shm_perm.gid和shm_perm.mode
3. 简单的程序
代码实现
common.h
#ifndef _COMMON_H_
#define _COMMON_H_
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#define FTOK_FILE "/home/delphi/ftok.file"
#define FTOK_ID 1
#define SHM_RD_PERMISSION 0444
#define SHM_WR_PERMISSION 0222
#define SHM_RW_PERMISSION (SHM_RD_PERMISSION | SHM_WR_PERMISSION)
#endif
shmcreate.c
#include "common.h"
int main(int argc, char **argv)
{
int length = atoi(argv[1]);
int oflag = IPC_CREAT | SHM_RW_PERMISSION;
int shmid = shmget(ftok(FTOK_FILE, FTOK_ID), length, oflag);
if (shmid >= 0)
{
printf("shmget create success, shmid = %d\n", shmid);
}
return 0;
}
shmrmid.c
#include "common.h"
int main(int argc, char **argv)
{
int shmid = shmget(ftok(FTOK_FILE, FTOK_ID), 0, SHM_RW_PERMISSION);
shmctl(shmid, IPC_RMID, NULL);
return 0;
}
shmwrite.c
#include "common.h"
int main(int argc, char **argv)
{
int shmid;
unsigned char *shmadd;
struct shmid_ds buf;
int i;
shmid = shmget(ftok(FTOK_FILE, FTOK_ID), 0, SHM_RW_PERMISSION);
shmadd = shmat(shmid, NULL, 0);
shmctl(shmid, IPC_STAT, &buf);
for (i = 0; i < buf.shm_segsz; i++)
{
*shmadd++ = i % 256;
}
return 0;
}
shmread.c
#include "common.h"
int main(int argc, char **argv)
{
int shmid;
unsigned char *shmadd;
unsigned char v;
struct shmid_ds buf;
int error = 0;
int i;
shmid = shmget(ftok(FTOK_FILE, FTOK_ID), 0, SHM_RW_PERMISSION);
shmadd = shmat(shmid, NULL, 0);
shmctl(shmid, IPC_STAT, &buf);
for (i = 0; i < buf.shm_segsz; i++)
{
v = *shmadd++;
if (v != (i % 256))
{
printf("error: shmadd[%d] = %d\n", i, v);
error++;
}
}
if (error == 0)
{
printf("all of read is ok\n");
}
return 0;
}
代码测试
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共享内存介绍
(一)简单概念 共享内存作为一种进程间通信的方式,其相较于其他进程间通信方式而言最大的优点就是数据传输速率快.其内部实现的方式采用了Linux进程地址空间中的mmap文件映射区,将文件内容直接映射到各 ...
- 共享内存之——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 ...
随机推荐
- flink入门(一)——基本原理与应用场景
一.简介 1.简介 flink是一个开源的分布式流处理框架 优势:高性能处理.高度灵活window操作.有状态计算的Exactly-once等 详情简介,参考官网:https://flink.apac ...
- 简单的 js 模版引擎
简单的 js 模版引擎 var tplEngine = function(tpl, data) { var reg = /<%([^%>]+)?%>/g, regOut = /(^( ...
- 【446】Deep Learning
ref: 深度学习基础介绍 机器学习19 神经网络NN算法 ref: 深度学习基础介绍 机器学习11 神经网络算法应用上 ref: 深度学习基础介绍 机器学习12 神经网络算法应用下 ref: 神经网 ...
- ELK 部署文档
1. 前言 在日常运维工作中,对于系统和业务日志的处理尤为重要.尤其是分布式架构,每个服务都会有很多节点,如果要手工一个一个的去取日志,运维怕是要累死. 简单介绍: ELK 是 elasticsear ...
- laravel composer vendor 目录加载类库详细 之后做说明
composer installLoading composer repositories with package informationInstalling dependencies (inclu ...
- Java项目main方法启动的两种方式
1.打包时指定了主类,可以直接用java -jar xxx.jar. <!--main方法打包jar包插件--> <plugin> <artifactId>mave ...
- java的Array和List相互转换
1.Array转List,通过java.util.Arrays.asList(T... a)参数是可变泛型参数 注意,Arrays.asList返回的类型是不可变长度的集合,底层是final修饰的泛型 ...
- linux awk的用法
linux awk的用法 <pre>[root@iZ23uewresmZ ~]# cat /home/ceshis.txtb 12 42 30 b 03 43 25 a 08 10 16 ...
- 面试必备的10道MySQL题
MySQL 事务,是我们去面试中高级开发经常会被问到的问题,很多人虽然经常使用 MySQL,SQL 语句也写得很溜,但是面试的时候,被问到这些问题,总是不知从何说起.下面我们先来了解一下什么是 MyS ...
- String字符串相关操作
.length 字符串长度.equals 比较字符串.equalIgnoreCase 比较字符串不区别大小写.charAt 获取字符串指定下标位置的字符.contains 判断字符串内是否包含某字符串 ...