进程间通信的方式

前言

进程间的通信包括三个方面:
分别是:

  1. 管道
  2. 系统IPC(消息队列、信号量、信号、共享内存)
  3. 套接字socket

一、管道

管道主要包括匿名管道命名管道,他是内存中的一个文件;

1、匿名管道

匿名管道可用于具有亲缘关系的父子进程间的通信;
1)它是半双工的(即数据只能在一个方向上流动),具有固定的读端和写端
2)它只能用于具有亲缘关系的进程之间的通信(也是父子进程或者兄弟进程之间)
3)它可以看成是一种特殊的文件,对于它的读写也可以使用普通的read、write等函数。但是它不是普通的文件,并不属于其他任何文件系统,并且只存在于内存中。

2、命名管道FIFO

命名管道除了具有管道所具有的功能外,它还允许无亲缘关系进程间的通信 ;
1)FIFO可以在无关的进程之间交换数据 ;
2)FIFO有路径名与之相关联,它以一种特殊设备文件形式存在于文件系统中。

二、系统IPC

2.1 消息队列

  • 消息队列,是消息的链接表,存放在内核中,独立于发送和接收进程;
  • 消息队列克服了信号传递信息少,管道只能承载无格式字节流以及缓冲区大小受限等特点;
  • 具有写权限得进程可以按照一定得规则向消息队列中添加新信息,有读权限得进程则可以从消息队列中读取信息;

特点:

  1. 消息队列是面向记录的,其中的消息具有特定的格式以及特定的优先级;
  2. 消息队列独立于发送与接收进程。进程终止时,消息队列及其内容并不会被删除;
  3. 消息队列可以实现消息的随机查询,消息不一定要以先进先出的次序读取,也可以按消息的类型读取。

2.2 信号量

信号量是一种机制,用于控制多个进程对共享资源的访问(互斥和同步),而不是内存中的存储通信数据的一块资源;

特点:

  1. 信号量用于进程间同步,若要在进程间传递数据需要结合共享内存;
  2. 信号量基于操作系统的 PV 操作,程序对信号量的操作都是原子操作;
  3. 每次对信号量的 PV 操作不仅限于对信号量值加 1 或减 1,而且可以加减任意正整数;
  4. 支持信号量组;

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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值