最近我们有一个需求,要求将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);
}
}
}
这个问题,其实就是要求能够 截获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);
}
}
}
本文介绍了一种在Windows控制台程序中将标准输入输出重定向到远程终端的方法,利用CreatePipe和SetStdHandle等API实现了数据的截获与TCP传输。
981

被折叠的 条评论
为什么被折叠?



