匿名管道是一种进程通信方式。
它有很大的局限性: 只能使用于本地,不能用于网络间进程通信; 只能适用于本地父子进程间的通信。但匿名管道可以实现控制台程序的重定向功能!对于多数常见的IDE比如VS,算法实现是一个控制台程序,运行时会将其运行信息重定向到GUI中。另外也可以通过SetStdHandle/GetStdHandle完成控制台到控制台的重定向。这不得不让人拍案叫绝,这是一种很好的设计思想。将实现与显示分离。
测试代码
下面给出一个测试代码:
功能描述:利用重定向,在界面中显示ping命名的过程。
开发环境:VS2008,MFC程序。
关键代码:
//显示 子进程重定向的输出
LRESULT CNPSevDlg::OnDataRec(WPARAM wParam, LPARAM lParam)
{
m_lstResult.AddString(m_szData);
return 0;
}
namespace {
//读管道线程
UINT ReadPipeProc( LPVOID pParam )
{
CNPSevDlg * pAttachWnd = static_cast<CNPSevDlg *>(pParam);
HANDLE hRead = pAttachWnd->m_hReadPipe;
HWND hWnd = pAttachWnd->GetSafeHwnd();
DWORD bytesRead;
while( 1 )
{
int len = sizeof(pAttachWnd->m_szData);
ZeroMemory(&pAttachWnd->m_szData,len);
if( !ReadFile(hRead, pAttachWnd->m_szData, len-1, &bytesRead, NULL) )
break;
SendMessage(hWnd,MSG_DATAREC,0,0);
}
return 0;
}
}
//启用ping命令行
void CNPSevDlg::OnBnClickedButton1()
{
HANDLE hRead, hWrite;
//创建匿名管道
{
SECURITY_ATTRIBUTES sa;
ZeroMemory(&sa,sizeof(sa));
sa.nLength = sizeof(SECURITY_ATTRIBUTES);
sa.lpSecurityDescriptor = NULL; //默认的安全描述符
sa.bInheritHandle = TRUE; //这个必须要设定TRUE,参考资料:《windows核心编程》第三章
if( !CreatePipe(&hRead, &hWrite, &sa, 0) )
{
MessageBox(" CreatePipe return FALSE.");
return;
}
m_hReadPipe = hRead;
}
//创建子进程
{
//设置启动参数
STARTUPINFO si;
PROCESS_INFORMATION pi;
si.cb = sizeof(STARTUPINFO);
GetStartupInfo(&si);
si.hStdError = hWrite;
si.hStdOutput = hWrite;
si.wShowWindow = SW_HIDE;
si.dwFlags = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES;
AfxBeginThread(ReadPipeProc,this,NULL);
TCHAR szCmdPar[256] = "ping www.baidu.com";
if(!CreateProcess(NULL, szCmdPar,NULL,NULL,TRUE,NULL,NULL,NULL,&si, &pi))
{
MessageBox("CreateProcess failed!");
return;
}
CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
}
}
运行效果: