学习分享
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