windows下printf重定向的实现

本文介绍了一种在Windows控制台程序中将标准输入输出重定向到远程终端的方法,利用CreatePipe和SetStdHandle等API实现了数据的截获与TCP传输。
最近我们有一个需求,要求将windows控制台程序中通过printf打印的字符串信息全部重定向到另一个远程控制终端上去;并且也将标准输入也重定向到远程终端。重定向信息的传输用TCP传输。解决
这个问题,其实就是要求能够 截获printf的字符数据,使其不在控制台上打印,而是通过socket传到
远程终端后打印。并且将标准输入重定向到一个socket上,从上面阻塞收数据作为标准输入。
解决过程:
1. 当然是到网上找答案了。的确发现有不少兄弟跟我们有同样的需求。搜到了与之相关的三种解决
方案。
   <1> 用freopen可以将标准输出可以重定向到一个打开的文件中。调用该函数后就可以将printf的
数据自动写入到一个文件中了。这种方法确实实现了标准输出的重定向,但是总不能让我反复的读文
件,再把数据通过socket发出去吧。。。
   <2> 用SetStdHandle重定向标准输入和输出。大家都找到了这个函数,表面看来通过它的确可以
很容易的将标准输入和输出重定向到一个管道中去。可是实践证明,我们被这个“美好”的接口给涮
了。msdn中给出了这个接口的一个示例,那个示例还挺麻烦的,让一个文件中的数据通过管道发送到
一个子进程中后再从子进程中通过管道发回来,绕了一圈最后打印到黑窗口控制台上。费了半天事把
那个demo看明白了。满怀欣喜的调用SetStdHandle把标准输出重定向到一个管道后,调了一把
printf("say hi to everybody\n"),结果还是显示到了黑窗口上,根本就没有往管道里送!愤怒,幸
好N多人都被它蒙了一把,心里才平衡一些。大家到这个时候一般就一筹莫展了。
   <3> http://www.regexlab.com/zh/stdredir/
      这个帖子中的哥们实现了一个RedirectStdout函数,可以重定向标准输出/输入/能通过callback函数截获数据…….帖子写的很好,可是那个哥们太吝啬了,不提供源码!在这里鄙视一下。把他的那个库下载下来,在vc.2003下用也一下,根本没有反应!!
2. 彷徨……
   实在是在网上找不到答案了,彷徨中。。。
3. 曙光……
  http://my.donews.com/wucr/2006/11/27/ipskeiovqkvhupzswvbxgqpwsiyjnezvurfz/
   这篇帖子里讲了一种在窗口程序中用控制台来显示调试打印的方法,其中的核心代码摘录如下
   HANDLE hOutput=GetStdHandle(STD_OUTPUT_HANDLE);
   HANDLE hInPut=GetStdHandle(STD_INPUT_HANDLE);
   //redirect stdout and stdin
   int hCrt;
   FILE *hf,*hf2;
   hCrt = _open_osfhandle((intptr_t)hOutput,0×4000);
   hf = _fdopen( hCrt, “w” );
   *stdout = *hf;
   hCrt = _open_osfhandle((intptr_t)hInPut, 0×4000);
   hf2 = _fdopen(hCrt,”r”);
   *stdin = * hf2; 
   写到这里大家应该明白了,要向重定向标准输入和输出还是要在CRT库中的stdout和stdin上做文章, 这才是根本。
4. 终极解决
   //*******************************************************************
   // 创建一个管道,用于重定向标准输出
   if (! CreatePipe(&g_hChildStdoutRd, &g_hChildStdoutWr, &saAttr, 0))
   {
       printf("Stdout pipe creation failed\n");
   }
   // 用3中提到的方法来将标准输出重定向到这个管道上
   hCrt = _open_osfhandle((long)g_hChildStdoutWr, 0x4000);
   hf = _fdopen( hCrt, "w" );
   *stdout = *hf;
   setvbuf(stdout,NULL,_IONBF,0);
   // 创建一个线程,从管道的另一头截获数据
   CreateThread(NULL,
                NB_TPF_SERVER_INPUT_COMMAND_TASK_STACK_SIZE,
                (LPTHREAD_START_ROUTINE)console_thread_out,
                (LPVOID)NULL,
                0,
                NULL);
   //***********************************************************************
   OSP_STATUS console_thread_out(void)
   {
      CHAR chBuf[1024];
      DWORD dwRead;
      BOOL fSuccess;
      while (1)
      {
         // 测试
         printf("say hi to everybody\n");
         // 从管道的另一头截获数据
         fSuccess = ReadFile(g_hChildStdoutRd, chBuf, 1024, &dwRead, NULL);
         // 用socket发到远程终端
         send(g_s32PrintMsgSockFd, chBuf, dwRead, 0); 
      }
   }
}
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值