前言
进程间的通信包括三个方面:
分别是:
- 管道
- 系统IPC(消息队列、信号量、信号、共享内存)
- 套接字socket
一、管道
管道主要包括匿名管道和命名管道,他是内存中的一个文件;
1、匿名管道
匿名管道可用于具有亲缘关系的父子进程间的通信;
1)它是半双工的(即数据只能在一个方向上流动),具有固定的读端和写端
2)它只能用于具有亲缘关系的进程之间的通信(也是父子进程或者兄弟进程之间)
3)它可以看成是一种特殊的文件,对于它的读写也可以使用普通的read、write等函数。但是它不是普通的文件,并不属于其他任何文件系统,并且只存在于内存中。
2、命名管道FIFO
命名管道除了具有管道所具有的功能外,它还允许无亲缘关系进程间的通信 ;
1)FIFO可以在无关的进程之间交换数据 ;
2)FIFO有路径名与之相关联,它以一种特殊设备文件形式存在于文件系统中。
二、系统IPC
2.1 消息队列
- 消息队列,是消息的链接表,存放在内核中,独立于发送和接收进程;
- 消息队列克服了信号传递信息少,管道只能承载无格式字节流以及缓冲区大小受限等特点;
- 具有写权限得进程可以按照一定得规则向消息队列中添加新信息,有读权限得进程则可以从消息队列中读取信息;
特点:
- 消息队列是面向记录的,其中的消息具有特定的格式以及特定的优先级;
- 消息队列独立于发送与接收进程。进程终止时,消息队列及其内容并不会被删除;
- 消息队列可以实现消息的随机查询,消息不一定要以先进先出的次序读取,也可以按消息的类型读取。
2.2 信号量
信号量是一种机制,用于控制多个进程对共享资源的访问(互斥和同步),而不是内存中的存储通信数据的一块资源;
特点:
- 信号量用于进程间同步,若要在进程间传递数据需要结合共享内存;
- 信号量基于操作系统的 PV 操作,程序对信号量的操作都是原子操作;
- 每次对信号量的 PV 操作不仅限于对信号量值加 1 或减 1,而且可以加减任意正整数;
- 支持信号量组;
2.3 信号
信号是一种比较复杂的通信方式,用于通知接收进程某个事件已经发生。
2.4 共享内存
它使得多个进程可以访问同一块内存空间,不同进程可以及时看到对方进程中对共享内存中数据得更新。这种方式需要依靠某种同步操作,如互斥锁和信号量等 ;
特点:
1.共享内存是最快的一种IPC,因为进程是直接对内存进行存取;
2. 但是由于是没有结构的同一资源 ,需要进行进程间的协同和互斥;
3. 信号量+共享内存通常结合在一起使用,信号量用来同步对共享内存的访问
三、socket
socket也是一种进程间远距离通信的机制,是真正的通信,依赖于TCP/IP协议,与其他‘‘通信’‘机制不同的是,它可用于不同主机之间的进程通信,不涉及互斥和同步的操作。
PS : 还是贴一段代码吧,匿名管道,
父进程:
#include <windows.h>
#include <iostream>
using namespace std;
int main(){
PROCESS_INFORMATION pi;
HANDLE hRead;//管道读句柄
HANDLE hWrite;//管道写句柄
BOOL bCSuc = CreatePipe(&hRead, &hWrite, NULL, 0);//创建匿名管道
if (bCSuc)
cout << "创建匿名管道成功." << endl;
else
cout << "创建匿名管道成功." << GetLastError();//出错原因
//记录主进程的标准输出
HANDLE hTemp = GetStdHandle(STD_OUTPUT_HANDLE);
//然后将输出写入匿名管道
SetStdHandle(STD_OUTPUT_HANDLE, &hWrite);
//获取主进程的STARTUPINFO结构信息
STARTUPINFO si;
GetStartupInfo(&si);//并写入si
//创建子进程
bCSuc = CreateProcess(
NULL, (LPSTR)"\"C:\\Users\\lenovo\\Desktop\\NMpiple\\Debug\\inp.exe\" -L -S",
NULL, NULL, FALSE, NULL, NULL, NULL,
&si, &pi);
SetStdHandle(STD_OUTPUT_HANDLE, hTemp); //恢复本进程的标准输出
if (bCSuc)
cout << "开始子进程成功" << endl;
else
cout << "开始子进程失败" << GetLastError() << endl;
CloseHandle(hWrite);//关闭这个进程写句柄
char ReadBuf[100];
DWORD ReadNum;
while (ReadFile(hRead, ReadBuf, 100, &ReadNum, NULL)){
ReadBuf[ReadNum] = '\0';
cout << "从管道读取" << ReadNum << "子节数据" << endl;
cout << ReadBuf << endl;
}
if (GetLastError() == ERROR_BROKEN_PIPE) // 输出信息
cout<<"管道已经被子进程关闭"<<endl;
else
cout<<"读数据错误,错误代码:"<< GetLastError()<<endl;
Sleep(1000);
system("pause");
return 0;
}
子进程
#include <iostream>
using namespace std;
int main(){
for (int i = 0; i<200; ++i){
printf("i=%d",i);
//cout << "a" << i << endl;
cerr << "b" << i << endl;
}
return 0;
}