【Linux基础(五)】IPC

1、进程间通信

进程间通信有如下一些目的:

数据传输:一个进程需要将它的数据发送给另一个进程,发送的数据量在一个字节到几兆字节之间。
共享数据:多个进程想要操作共享数据,一个进程对共享数据的修改,别的进程应该立刻看到。
通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止时要通知父进程)。
资源共享(锁):多个进程之间共享共同的资源。为了作到这一点,需要内核提供锁和同步机制。
进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变。

2、IPC发展

linux进程间通信(IPC)由以下几部分发展而来
早期UNIX进程间通信、基于System V进程间通信、基于Socket进程间通信和POSIX进程间通信。
UNIX进程间通信方式包括:管道、FIFO、信号
System V进程间通信方式包括:System V消息队列、System V信号灯、System V共享内存。
POSIX进程间通信包括:posix消息队列、posix信号灯、posix共享内存。

3、当前IPC技术

现在Linux使用的进程间通信方式
(1)管道(pipe)和命名管道(FIFO)
(2)信号(signal)
(3)消息队列
(4)共享内存
(5)信号量
(6)套接字(socket)

3.1、消息队列

消息队列提供了从一个进程向另外一个进程发送一块数据的方法
每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值
消息队列也有管道一样的不足,就是每个数据块的最大长度是有上限的,系统上全体队列的最大总长度也有一个上限

消息队列是消息的链表,存放在内核中并由消息队列标识符表示。内核为每个IPC对象维护了一个数据结构struct ipc_perm,用于标识消息队列,让进程知道当前操作的是哪个消息队列。每一个msqid_ds表示一个消息队列,并通过msqid_ds.msg_first、msg_last维护一个先进先出的msg链表队列,当发送一个消息到该消息队列时,把发送的消息构造成一个msg的结构对象,并添加到msqid_ds.msg_first、msg_last维护的链表队列

在这里插入图片描述

3.1.1、msgget函数

在这里插入图片描述

3.1.2、msgsnd函数

在这里插入图片描述
在这里插入图片描述

3.1.3、msgrcv函数

在这里插入图片描述
在这里插入图片描述

3.1.4、消息队列–发送者示例

#include <iostream>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdio.h>
using namespace std;

typedef struct messagebuf {
	long mtype;       /* message type, must be > 0 */
	char mtext[10];    /* message data */
}MSGBUF;

int main()
{	
	int msgid = msgget((key_t)1001,IPC_CREAT|0777);
	if (msgid < 0)
	{
		perror("msgget error");
	}
	else
	{
		MSGBUF buf;
		buf.mtype = 1;
		sprintf(buf.mtext, "Hello%d", 1);
		if (msgsnd(msgid, &buf, sizeof(buf.mtext), 0) < 0)
		{
			perror("msgsnd error");
		}
		else
		{
			cout << "消息发送成功" << endl;
		}
	}
	return 0;
}

3.1.5、消息队列–接收者示例

#include <iostream>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdio.h>
using namespace std;

typedef struct messagebuf {
	long mtype;       /* message type, must be > 0 */
	char mtext[10];    /* message data */
}MSGBUF;

int main()
{
	int msgid = msgget((key_t)1001, IPC_CREAT | 0777);
	if (msgid < 0)
	{
		perror("msgget error");
	}
	else
	{
		MSGBUF buf;
		//阻塞式函数
		if (msgrcv(msgid, &buf, sizeof(buf.mtext),1, 0) < 0)
		{
			perror("msgrcv error");
		}
		else
		{
			cout << "接收消息成功" << endl;
			cout << "接收到消息 buf.mtext=" << buf.mtext << endl;
		}
	}
	return 0;
}

3.2、共享内存

共享内存允许两个不相关的进程去访问同一部分逻辑内存
如果需要在两个运行中的进程之间传输数据,共享内存将是一种效率极高的解决方案

在这里插入图片描述
在这里插入图片描述

3.2.1、shmget函数

在这里插入图片描述

3.2.2、shmat函数

在这里插入图片描述
在这里插入图片描述

在fork() 后,子进程继承已连接的共享内存
在exec后,已连接的共享内存会自动脱离(detach)
在结束进程后,已连接的共享内存会自动脱离(detach)

3.2.3、shmdt函数

在这里插入图片描述

3.2.4、共享内存–写入数据

#include <iostream>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <string.h>
using namespace std;
typedef struct student
{
	char stuID[20];
	char stuName[20];
}STU;
int main()
{
	void* shmaddr = NULL;
	//创建共享内存
	int shmid = shmget((key_t)1002, sizeof(STU) * 3, IPC_CREAT | 0777);
	if (shmid < 0)
	{
		perror("shmget error");
	}
	else
	{
		//连接共享内存  找到共享内存首地址
		shmaddr = shmat(shmid, NULL, 0);
		STU stu1 = { "1001","张三" };
		STU stu2 = { "1002","李四" };
		STU stu3 = { "1003","王五" };
		memcpy(shmaddr, &stu1, sizeof(STU));
		memcpy(shmaddr + sizeof(STU), &stu2, sizeof(STU));
		memcpy(shmaddr + sizeof(STU) * 2, &stu3, sizeof(STU));
		cout << "拷贝数据成功" << endl;
		//断开连接
		shmdt(shmaddr);
	}
	return 0;
}

3.2.5、共享内存–读取数据

#include <iostream>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <string.h>
using namespace std;
typedef struct student
{
	char stuID[20];
	char stuName[20];
}STU;
int main()
{
	void* shmaddr = NULL;//创建共享内存
	int shmid = shmget((key_t)1002, sizeof(STU) * 3, IPC_CREAT | 0777);
	if (shmid < 0)
	{
		perror("shmget error");
	}
	else
	{
		//连接共享内存  找到共享内存首地址
		shmaddr = shmat(shmid, NULL, 0);
		STU stu;
		//参数1:目的地  参数2:始发地   参数3 始发地发送大小
		memcpy(&stu, shmaddr, sizeof(STU));
		cout << "ID-1- " << stu.stuID << endl;
		cout << "Name-1- " << stu.stuName << endl;

		memcpy(&stu, shmaddr + sizeof(STU), sizeof(STU));
		cout << "ID-2- " << stu.stuID << endl;
		cout << "Name-2- " << stu.stuName << endl;

		memcpy(&stu, shmaddr + sizeof(STU) * 2, sizeof(STU));
		cout << "ID-3- " << stu.stuID << endl;
		cout << "Name-3- " << stu.stuName << endl;

		memset(shmaddr, 0, sizeof(STU) * 3);
		cout << "清空共享内存" << endl;
		//断开连接
		shmdt(shmaddr);
	}
	return 0;
}

4、ipcs–显示消息列表、共享内存和信号量信息

ipcs命令用于报告 Linux 中进程间通信设施的状态,显示的信息包括消息列表、共享内存和信号量的信息。(这三个独立于进程之外)

在这里插入图片描述

5、 ipcrm–删除共享内存、消息队列,信号量

ipcrm -l

在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值