可以使用管道

Visual C++下对匿名管道的编程实现

概述

  管道(Pipe)实际是用于进程间通信的一段共享内存,创建管道的进程称为管道服务器,连接到一个管道的进程为管道客户机。一个进程在向管道写入数据后,另一进程就可以从管道的另一端将其读取出来。匿名管道(Anonymous Pipes)是在父进程和子进程间单向传输数据的一种未命名的管道,只能在本地计算机中使用,而不可用于网络间的通信。

  匿名管道实施细则

  匿名管道由CreatePipe()函数创建,该函数在创建匿名管道的同时返回两个句柄:管道读句柄和管道写句柄。CreatePipe()的函数原型为:

BOOL CreatePipe(PHANDLE hReadPipe, // 指向读句柄的指针
 PHANDLE hWritePipe, // 指向写句柄的指针
 LPSECURITY_ATTRIBUTES lpPipeAttributes, // 指向安全属性的指针
 DWORD nSize // 管道大小
);

  通过hReadPipe和hWritePipe所指向的句柄可分别以只读、只写的方式去访问管道。在使用匿名管道通信时,服务器进程必须将其中的一个句柄传送给客户机进程。句柄的传递多通过继承来完成,服务器进程也允许这些句柄为子进程所继承。除此之外,进程也可以通过诸如DDE或共享内存等形式的进程间通信将句柄发送给与其不相关联的进程。

  在调用CreatePipe()函数时,如果管道服务器将lpPipeAttributes 指向的SECURITY_ATTRIBUTES数据结构的数据成员bInheritHandle设置为TRUE,那么CreatePipe()创建的管道读、写句柄将会被继承。管道服务器可调用DuplicateHandle()函数改变管道句柄的继承。管道服务器可以为一个可继承的管道句柄创建一个不可继承的副本或是为一个不可继承的管道句柄创建一个可继承的副本。CreateProcess()函数还可以使管道服务器有能力决定子进程对其可继承句柄是全部继承还是不继承。

  在生成子进程之前,父进程首先调用Win32 API SetStdHandle()使子进程、父进程可共用标准输入、标准输出和标准错误句柄。当父进程向子进程发送数据时,用SetStdHandle()将管道的读句柄赋予标准输入句柄;在从子进程接收数据时,则用SetStdHandle()将管道的写句柄赋予标准输出(或标准错误)句柄。然后,父进程可以调用进程创建函数CreateProcess()生成子进程。如果父进程要发送数据到子进程,父进程可调用WriteFile()将数据写入到管道(传递管道写句柄给函数),子进程则调用GetStdHandle()取得管道的读句柄,将该句柄传入ReadFile()后从管道读取数据。

  如果是父进程从子进程读取数据,那么由子进程调用GetStdHandle()取得管道的写入句柄,并调用WriteFile()将数据写入到管道。然后,父进程调用ReadFile()从管道读取出数据(传递管道读句柄给函数)。

  在用WriteFile()函数向管道写入数据时,只有在向管道写完指定字节的数据后或是在有错误发生时函数才会返回。如管道缓冲已满而数据还没有写完,WriteFile()将要等到另一进程对管道中数据读取以释放出更多可用空间后才能够返回。管道服务器在调用CreatePipe()创建管道时以参数nSize对管道的缓冲大小作了设定。

  匿名管道并不支持异步读、写操作,这也就意味着不能在匿名管道中使用ReadFileEx()和WriteFileEx(),而且ReadFile()和WriteFile()中的lpOverLapped参数也将被忽略。匿名管道将在读、写句柄都被关闭后退出,也可以在进程中调用CloseHandle()函数来关闭此句柄。
总的来说,匿名管道程序是比较简单的。在下面将要给出的程序示例中,将由父进程(管道服务器)创建一个子进程(管道客户机),子进程回见个其全部的标准输出发送到匿名管道中,父进程再从管道读取数据,一直到子进程关闭管道的写句柄。其中,匿名管道服务器程序的实现清单如下:

STARTUPINFO si;
PROCESS_INFORMATION pi;
char ReadBuf[100];
DWORD ReadNum;
HANDLE hRead; // 管道读句柄
HANDLE hWrite; // 管道写句柄
BOOL bRet = CreatePipe(&hRead, &hWrite, NULL, 0); // 创建匿名管道
if (bRet == TRUE)
 printf("成功创建匿名管道!/n");
else
 printf("创建匿名管道失败,错误代码:%d/n", GetLastError());
 // 得到本进程的当前标准输出
 HANDLE hTemp = GetStdHandle(STD_OUTPUT_HANDLE);
 // 设置标准输出到匿名管道
 SetStdHandle(STD_OUTPUT_HANDLE, hWrite);
 GetStartupInfo(&si); // 获取本进程的STARTUPINFO结构信息
 bRet = CreateProcess(NULL, "Client.exe", NULL, NULL, TRUE, NULL, NULL, NULL, &si, &pi); // 创建子进程
 SetStdHandle(STD_OUTPUT_HANDLE, hTemp); // 恢复本进程的标准输出
 if (bRet == TRUE) // 输入信息
  printf("成功创建子进程!/n");
 else
  printf("创建子进程失败,错误代码:%d/n", GetLastError());
  CloseHandle(hWrite); // 关闭写句柄
  // 读管道直至管道关闭
  while (ReadFile(hRead, ReadBuf, 100, &ReadNum, NULL))
  {
   ReadBuf[ReadNum] = '/0';
   printf("从管道[%s]读取%d字节数据/n", ReadBuf, ReadNum);
  }
  if (GetLastError() == ERROR_BROKEN_PIPE) // 输出信息
   printf("管道被子进程关闭/n");
  else
   printf("读数据错误,错误代码:%d/n", GetLastError());

  在本示例中,将当前进程的标准输出设置为使用匿名管道,再创建子进程,子进程将继承父进程的标准输出,然后再将父进程的标准输出恢复为其初始状态。于是父进程便可从管道读取数据,直到有错误发生或关闭管道写入端的所有句柄。创建的子进程只是向标准输出和标准错误发送一些文本信息,其中发送给标准输出的文本将重定向输出到管道,发送给标准错误的文本将不改变输出。下面给出子进程的实现代码:

int main(int argc, char* argv[])
{
 for (int i = 0; i < 100; i++) // 发送一些数据到标准输出和标准错误
 {
  printf("i = %d/n", i); // 打印提示
  cout << "标准输出:" << i << endl; // 打印到标准输出
  cerr << "标准错误:" << i << endl; // 打印到标准错误
 }
 return 0;
}

Visual C++下对匿名管道的编程实现相关推荐

  1. [C++] 匿名管道的理解与实现

    什么是匿名管道? 匿名管道用于进程之间通信,且仅限于本地父子进程之间通信,结构简单,类似于一根水管,一端进水另一端出水(单工).相对于命名管道,其占用小实现简单,在特定情况下,比如实现两围棋引擎本地对 ...

  2. Linux系统编程:pipe匿名管道的使用,实现linux命令下管道命令

    pipe函数介绍 函数原型int pipe(int pipefd[2]) 来创建匿名管道; 传出2个fd 文件描述符,pipefd[0]表示匿名管道的读端,pipefd[1]表示匿名管道的写端.有这个 ...

  3. 【Linux系统编程学习】匿名管道pipe与有名管道fifo

    此为牛客Linux C++和黑马Linux系统编程课程笔记. 0. 关于进程通信 Linux环境下,进程地址空间相互独立,每个进程各自有不同的用户地址空间.任何一个进程的全局变量在另一个进程中都看不到 ...

  4. Linux系统编程- 无名管道(匿名管道)

    无名管道作为Linux进程间通讯,我们这里把理论和实际结合起来说明. 1.什么是管道 管道,英文位pipe,在学习linux系统编程一个重要概念.它的发明人是道格拉斯.麦克罗伊,这位也是UNIX上早期 ...

  5. C++学习:第六章Linux高级编程 - (七)信号、sigqueue、sigaction、IPC、管道、匿名管道

    回顾: 1. 信号的作用 2. 理解信号: 软中断 可靠与不可靠信号 kill -l 3. 信号发送与注册 kill/raise alarm setitimer signal 4. 信号的屏蔽 sig ...

  6. 匿名管道 阻塞_Linux系统编程—有名管道

    ▋**1. 管道的概念** 管道,又名「无名管理」,或「匿名管道」,管道是一种非常基本,也是使用非常频繁的IPC方式. 1.1 管道本质 管道的本质也是一种文件,不过是伪文件,实际上是一块内核缓冲区, ...

  7. Linux系统编程27:进程间通信之管道的基本概念和匿名管道与命名管道及管道特性

    文章目录 (1)管道是什么 (2)匿名管道 A:读端和写端 B:建立匿名管道的函数 C:最简单的进程间通信-演示 D:管道四大特性 E:管道的特点 F:从内核角度理解管道 G:管道总结 (3)命名管道 ...

  8. Linux编程:多个子进程与父进程使用匿名管道读写

    编写程序,在程序中父进程先后创建子进程1和子进程2.父子三个进程并发执行,并利用同一个匿名管道通信:两个子进程分别向管道写一则信息(内容自定,但要求包含子进程信息):父进程从管道读出信息并显示出来(要 ...

  9. C/C++编程 获取调用其他程序的输出、匿名管道读写

    用途 调用外部命令来协助完成某些功能 如很多扫描工具实际上是调用nmap来帮助扫描任务的 #include <stdio.h> //#include <string.h> #i ...

最新文章

  1. unity windows打包ios_ios打包unity应用以及配置签名!
  2. 王高利:awstats
  3. Java 8特性有望进入.Net/Mono
  4. mysql 选择特定的表_MySQL选择具有多个特定列的所有表
  5. 网络设计分层设计的原理
  6. php curl nginx post 空_【青藤云安全研究】绕过php的disable_functions(上篇)
  7. python 网络爬虫 第一天
  8. 和opengl的关系_从零开始的图形学学习(零):一切的开始 —— 自建OpenGL开发框架...
  9. [通用]汉字按照拼音字母排序
  10. 如何安装王码五笔字型输入法86版
  11. 配置maven使用阿里云仓库
  12. 数据采集卡的模拟输入信号到底应该怎么接线
  13. Navicat无法导入excel文件的异常处理
  14. python中str是什么函数_python里的str是什么函数
  15. 数据分析(7)路径挖掘分析法 行为序列分析法
  16. java 怎么让图片运动,小编给你传授java怎么实现键盘控制图片移动
  17. 音乐手记之民谣二:Chamber
  18. 华为2019秋招面试问答题!(附带笔试参考题)
  19. 高数--函数--初等函数
  20. Struts2报错Caused by: java.lang.NoSuchMethodException: bean.Student.init()

热门文章

  1. UML图系列——UML模型图的构成
  2. Activiti工作流之实现一个简单的流程审批
  3. 类型转换:隐式转化(算数转换,整型提升,混合提升,赋值转换),强制转换【C语言】
  4. archlinux 开机自动连接wifi
  5. 学计算机趣图,我的世界:六张玩家自制趣图,最后一张,想起了“骗”父母买电脑...
  6. bigdecimal取小数部分_Java中BigDecimal保留两位小数点有哪些方法
  7. 深度解读 OpenYurt:从边缘自治看 YurtHub 的扩展能力
  8. Helm 从入门到实践 | 从 0 开始制作一个 Helm Charts
  9. 正式开放 | 阿里云 10 亿级镜像服务正式支持 Helm Charts,云原生交付再加速!
  10. 戴尔硬盘保护增强套件_拆解戴尔服务器,看看内部构造与普通计算机的区别