linux系统编程之共享内存

一、【共享内存的概述】

共享内存允许两个或者多个进程共享指定的存储区域

共享内存的特点:

1、共享内存是进程间共享数据的一种最快的方法。一个进程向共享的内存区域写入了数据,共享这个内存区域的所有进程就可以立刻看到其中的内容。

2、使用共享内存要注意的是多个进程之间对一个给定存储区访问的互斥若一个进程正在向共享内存区写数据,则在它做完这一步操作前,别的进程不应当去读、写这些数据

二、【共享内存的API】

1、创建共享内存区shmget()函数

头文件

#include <sys/ipc.h>

#include <sys/shm.h>

函数原型

int shmget(key_t key, size_t size, int shmflg);

这里需要获取key,我们要先使用ftok函数去获取唯一的key值。

key_t key = ftok("/",xxxx);

功能

创建或打开一块共享内存区

参数:

key: IPC键值 需要使用ftok函数去获取

size:该共享存储段的长度(字节)

shmflg:标识函数的行为及共享内存的权限:

IPC_CREAT:如果不存在就创建

IPC_EXCL:如果已经存在则返回失败

位或权限位:共享内存位或权限位后可以设置共享内存的访问权限,格式和open函数的mode_t一样,但可执行权限未使用

返回值:

成功:返回共享内存标识符

失败:返回-1

 

2、共享区映射shmat函数

头文件:

#include <sys/types.h>

#include <sys/shm.h>

函数原型:

void *shmat(int shmid, const void *shmaddr, int shmflg);

功能:

将一个共享内存映射到调用进程的数据段中。

参数:

shmid :共享内存标识符

shmaddr:共享内存映射地址:如果是NULL就系统自动指定。

shmflg:.共享内存段的访问权限和映射条件:

0:共享内存具有可读可写权限。

SHM_RDONLY:只读。

SHM_RND: (shmaddr非空时才有效)

没有指定.SHM.RND则此段连接到shmaddr.所指定的地址上( shmaddn必须页对齐)。

指定工.SHM_RND: 则此段连接到shmaddr- shmaddn%SHMLBA 所表示的地址上

返回值:

成功:返回共享内存段映射地址

失败:.返回-1

3、解除共享区映射区shmdt函数

头文件:

#include <sys/types.h>

#include <sys/shm.h>

函数原型:

int shmdt(const void *shmaddr);

功能:

将共享内存和当当前进程分离(仅仅是断开联系并不删除共享内存)。

参数:

shmaddr:共享内存映射地址。

返回值:

成功返回0

失败返回-1。

4、控制共享内存shmctl函数

头文件:

#include <sys/ipc.h>↓

#include <sys/shm.h>↓

函数原型:

int shmctl(int shmid, int cmd,struct shmid_ds*buf) ;↓

功能:

共享内存空间的控制。

参数:

shmid:.共享内存标识符。

cmd:函数功熊的控制。

buf:shmid_ds数据类型的地址,用来存放或修改共享内存的属性

cmd:函数功能的控制

IPC_RMID:删除。

IPC_SET:设置shmid_ds参数

IPC_STAT:保存shmid_ds参数

SHM_LOCK:锁定共享内存段(超极用户)

SHM_UNLOCK:解锁共享内存段。

返回值:

成功返回0,失败返回-1

三、案例:创建共享内存,一个进程负责读,一个进程负责写

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include<string.h>
#include <unistd.h>
#include <sys/ipc.h>
#include<wait.h>

int main(int argc, char const *argv[])
{
	pid_t pid = fork();
	if(pid==-1)
	{
		perror("fork");
		return -1;
	}
	//子进程负责写
	if(pid==0)
	{
		//获取key值
		key_t key = ftok("/",2098);

		//创建共享内存
		int shm_id = shmget(key,64,IPC_CREAT|0666);

		//共享内存映射
		char *buf = (char*)shmat(shm_id,NULL,0);

		strcpy(buf,"hello yz1!");

		shmdt(buf);


	}
	//父进程负责收
	if(pid>0)
	{
		//获取key值
		key_t key = ftok("/",2098);

		//创建共享内存
		int shm_id = shmget(key,64,IPC_CREAT|0666);

		//共享内存映射
		char *buf = (char*)shmat(shm_id,NULL,0);

		printf("父进程:打印buf内容:%s\n",buf);
		
		shmdt(buf);
        wait(NULL);

	}
	return 0;
}

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux共享内存编程是指在Linux操作系统中使用共享内存实现进程间通信的编程技术。共享内存是一种特殊的内存区域,它可以被多个进程同时访问,从而实现进程间数据的共享。 在Linux中,使用共享内存需要经过以下几个步骤: 1. 创建共享内存区域:使用shmget()函数创建一个共享内存区域,并返回该区域的标识符。 2. 连接共享内存区域:使用shmat()函数将进程与共享内存区域连接起来,并返回连接后的内存地址。 3. 使用共享内存:进程可以使用连接后的内存地址读写共享内存中的数据。 4. 分离共享内存区域:使用shmdt()函数将进程与共享内存区域分离,使得进程不能再访问该内存区域。 5. 删除共享内存区域:使用shmctl()函数删除共享内存区域。 下面是一个使用共享内存实现进程间通信的例子: ``` #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/ipc.h> #include <sys/shm.h> #define SHM_SIZE 1024 int main() { int shmid; key_t key; char *shm, *s; // 生成共享内存key值 key = ftok(".", 'a'); // 创建共享内存区域 if ((shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0666)) < 0) { perror("shmget"); exit(1); } // 连接共享内存区域 if ((shm = shmat(shmid, NULL, 0)) == (char *) -1) { perror("shmat"); exit(1); } // 写入共享内存数据 s = shm; for (char c = 'a'; c <= 'z'; c++) { *s++ = c; } *s = '\0'; // 分离共享内存区域 if (shmdt(shm) == -1) { perror("shmdt"); exit(1); } // 重新连接共享内存区域 if ((shm = shmat(shmid, NULL, 0)) == (char *) -1) { perror("shmat"); exit(1); } // 读取共享内存数据 printf("Shared memory content: "); for (s = shm; *s != '\0'; s++) { putchar(*s); } putchar('\n'); // 分离共享内存区域 if (shmdt(shm) == -1) { perror("shmdt"); exit(1); } // 删除共享内存区域 if (shmctl(shmid, IPC_RMID, 0) == -1) { perror("shmctl"); exit(1); } return 0; } ``` 这个例子中,程序通过ftok()函数生成一个共享内存的key值,然后使用shmget()函数创建共享内存区域,使用shmat()函数将进程与共享内存区域连接起来,然后向共享内存中写入数据,再使用shmdt()函数分离共享内存区域,重新连接共享内存区域并读取数据,最后使用shmctl()函数删除共享内存区域。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值