进程间通信之共享内存

1.概念:

共享内存就是多个进程的地址空间映射到同一个物理内存,多个进程都能看到这块物理内存,共享内存可以提供给服务器进程和客户进程之间进行通信,不需要进行数据的复制,所以速度最快。


2.共享内存操作需要的函数:

(1)我们需要利用ftok函数生成key标识符。

key_t ftok(const char *pathname,int proj_id).


(2)我们使用shmgt函数,创建一个共享内存块,返回这个共享内存块的标识符shmid。

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

size是需要申请的共享内存的大小,需要注意的是,操作系统为你提供的大小的时候是按页来提供,所以size为4k的整数倍,

shmflg:如果要创建新的共享内存,那么就使用IPC_CREAT,IPC_EXCL,如果是已经存在的,那么只需要使用IPC_CREAT。


(3)用shmat挂接共享内存(将进程地址空间挂接到共享内存,共享内存是物理空间,可以有多个挂接)

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

shmid是挂接的进程号,

shmaddr置为NULL,让系统选择一个合适的地址空间进行挂接

shmflg表示什么方式进行挂接,一般都是取0.

函数返回各个进程挂接的虚拟的地址空间。


(4)用shmdt去挂接。

int shmdt(const void *shmaddr);


(5)用shmctl销毁共享内存

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

cmd取IPC_RMID表示删除这块共享内存

buf一般设置为NULL,不关心这个东西,消息队列中也有这么一个类似的结构体也是设置为NULL.


3.代码实现:

下面的例子是服务器进程进行创建共享内存,然后挂接,客户进程也挂接,然后服务器进程每2秒写入一个字符,

客户进程也接收到,打印,实现内存共享

comm.h

#ifndef _COMM_H_
#define _COMM_H_


#include<stdio.h>
#include<sys/shm.h>
#include<errno.h>
#include<sys/types.h>
#include<unistd.h>

#define PATHNAME "."
#define PROJ_ID 0x6666

int creat_shm(size_t size);
int get_shm();
int destroy_shm(int shmid);



#endif 

comm.c

#include"comm.h"

int common(size_t size,int flags)
{
	key_t key = ftok(PATHNAME,PROJ_ID);
	if (key < 0)
	{
		perror("ftok");
		return -1;
	}
	int shmid = shmget(key,size,flags);//得到共享内存的标识符shmid。
	if (shmid < 0)
	{
		perror("shmget");
		return -2;
	}
	return shmid;
}

int creat_shm(size_t size)
{
	return common(size,IPC_CREAT|IPC_EXCL|0x666);
}

int get_shm()
{
	return common(0,IPC_CREAT);
}

int destroy_shm(int shmid)
{
	if(shmctl(shmid,IPC_RMID,NULL)<0)
	{
		perror("shmctl");
		return -3;
	}
	return 0;
}
server.c

#include"comm.h"

int main()
{
	int shmid = creat_shm(4096);
	printf("shmid = %d\n",shmid);
	sleep(3);
	char *buf = shmat(shmid,NULL,0);//挂接//shmat第二个参数为空,表示系统分配地址空间,返回一个虚拟地址空间。
	printf("buf = %p\n",buf);
	int i = 0;

	while (1)
	{
		buf[i] = 'A'+i%26;
	i++;
		buf[i] = 0;
		sleep(2);
	}
	shmdt(buf); //去挂接。
	sleep(3);
	destroy_shm(shmid);

	return 0;
}
client.c

#include"comm.h"


int main()
{
	int shmid = get_shm();
	sleep(3);
	char *buf = shmat(shmid,NULL,0);//新的进程挂接到那块共享内存空间,能看到共享内存中的东西。
	
	while (1)
	{
		printf("%s\n",buf);
		sleep(2);
	}
	shmdt(buf);
	return 0;
}








在Linux下,进程间通信的一种方式是通过共享内存来实现的。共享内存允许两个或多个进程共享一定的存储区,这样它们就可以直接访问同一块内存区域,而不需要进行数据的复制。共享内存是一种高效的进程间通信方式,因为数据直接写入内存,不需要多次数据拷贝,所以传输速度很快\[2\]。 在使用共享内存进行进程间通信时,需要给共享内存创建一个唯一的身份ID,以便区分不同的共享内存。当进程需要访问共享内存时,需要在映射时带上这个ID,这样就可以确定访问的是哪一个共享内存\[3\]。 需要注意的是,共享内存并没有提供同步机制,也就是说,在一个进程结束对共享内存的写操作之前,并没有自动机制可以阻止另一个进程开始对它进行读取。为了实现多个进程对共享内存的同步访问,通常会使用信号量来实现对共享内存的同步访问控制\[2\]。 总结起来,Linux下的共享内存是一种高效的进程间通信方式,允许多个进程共享一块存储区。通过给共享内存创建唯一的身份ID,可以区分不同的共享内存。然而,共享内存并没有提供同步机制,需要使用信号量来实现对共享内存的同步访问控制\[2\]\[3\]。 #### 引用[.reference_title] - *1* *3* [Linux多进程间通信——共享内存实现](https://blog.youkuaiyun.com/zhm1949/article/details/124909541)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [Linux下进程间通信方式——共享内存](https://blog.youkuaiyun.com/xujianjun229/article/details/118584955)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值