Windows管道与StdIn、StdOut重定向

       stdin是标准输入,stdout是标准输出,stderr是标准错误输出。大多数的命令行程序从stdin输入,输出到stdout或stderr,有时我们需要重定向stdout,stderr,stdin。比如:将输出写入文件,又或者我们要将命令行程序输出结果显示到Windows对话框中。

      在Windows编程中,重定向需要用到管道(Pipe)的概念。管道是一种用于在进程间共享数据的机制。一个管道类似于一个管子的两端,一端是写入的,一端是读出的。由一个进程从写入端写入、另一个进程从读出端读出,从而实现通信,就向一个“管道”一样。

重定向的原理是:

首先声明两个概念:主程序(重定向的操纵者)、子进程(被重定向的子进程)

  • 如果要重定向stdout的话,先生成一个管道, 管道的写入端交给子进程去写,主程序从管道的读出端读数据,然后可以把数据写成文件、显示等等。重定向stderr和stdout是相同的。
  • 同理,要重定向stdin的话,生成一个管道, 管道的写入端由主程序写,子进程从管道的读出端读数据。

下面是一个输出重定向的例子:将一个命令行Ping的结果展示在编辑框里。


核心代码如下:

#define MSG_DATAREC WM_USER +0x400

BEGIN_MESSAGE_MAP(CStudyStdOut_RedirectGuiDlg, CDialog)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	//}}AFX_MSG_MAP
	ON_BN_CLICKED(IDC_BUTTON1, &CStudyStdOut_RedirectGuiDlg::OnBnClickedButton1)
	ON_MESSAGE(MSG_DATAREC,&CStudyStdOut_RedirectGuiDlg::OnDataRec)
END_MESSAGE_MAP()



void CStudyStdOut_RedirectGuiDlg::OnBnClickedButton1()
{
	// TODO: Add your control notification handler code here
	SECURITY_ATTRIBUTES sa;  
	ZeroMemory(&sa,sizeof(sa));  
	sa.nLength = sizeof(SECURITY_ATTRIBUTES);  
	sa.lpSecurityDescriptor = NULL;  //默认的安全描述符   
	sa.bInheritHandle = TRUE;        //这个必须要设定TRUE,参考资料:《windows核心编程》第三章   
	HANDLE hRead, hWrite;

	if( !CreatePipe(&hRead, &hWrite, &sa, 0) )  
	{  
		MessageBox(" CreatePipe return FALSE.");  
		return;  
	}  

	m_hReadPipe = hRead;
	AfxBeginThread(ReadPipeProc,this,NULL);
	  
	STARTUPINFO siStartInfo;
	PROCESS_INFORMATION piProcInfo;
	ZeroMemory(&siStartInfo,sizeof(STARTUPINFO));
	siStartInfo.cb =sizeof(STARTUPINFO);
	siStartInfo.dwFlags  |= STARTF_USESHOWWINDOW;
	siStartInfo.dwFlags  |= STARTF_USESTDHANDLES;
	siStartInfo.hStdOutput = hWrite;
	siStartInfo.hStdError  = hWrite;

	TCHAR szCmdPar[256] = "ping www.baidu.com";  
	if(!CreateProcess(NULL, szCmdPar,NULL,NULL,TRUE,NULL,NULL,NULL,&siStartInfo, &piProcInfo))  
	{  
		MessageBox("CreateProcess failed!");  
		return;  
	}

	CloseHandle(piProcInfo.hProcess);
	CloseHandle(piProcInfo.hThread);

}

UINT ReadPipeProc( LPVOID pParam )  
{  
	CStudyStdOut_RedirectGuiDlg * pAttachWnd = static_cast<CStudyStdOut_RedirectGuiDlg *>(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;  
}  
HRESULT CStudyStdOut_RedirectGuiDlg::OnDataRec( WPARAM wParam, LPARAM lParam )
{
	m_strEdit += m_szData;
	UpdateData(FALSE);
	return 1;
}

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要允许用户使用管道操作符“|”来同时执行两个进程,并把第一个进程的stdout重定向为第二个进程的stdin,你可以按照以下步骤进行: 1. 读取用户的输入,使用C语言中的fgets函数读取用户的输入,并将其存储在一个字符数组中。 2. 解析用户的输入,使用C语言中的strtok函数将用户输入的命令和参数分离开来。如果用户输入了管道操作符“|”,则将命令和参数分别存储在两个数组中。 3. 创建一个管道,使用C语言中的pipe函数创建一个管道,以便将第一个进程的stdout重定向为第二个进程的stdin。 4. 创建两个子进程,分别执行用户输入的两个命令,使用C语言中的fork函数创建两个子进程,并使用C语言中的exec函数执行用户输入的命令。在第一个子进程中,将stdout重定向管道的写入端;在第二个子进程中,将stdin重定向管道的读取端。 5. 关闭管道和子进程,等待子进程结束,关闭管道的读取和写入端。 下面是一个简单的示例代码: ``` #include <stdio.h> #include <string.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/wait.h> int main() { char input[100]; char *command1, *command2; char *args1[10], *args2[10]; int i, pipefd[2]; while (1) { // 读取用户输入 printf("myshell> "); fgets(input, sizeof(input), stdin); // 去掉输入字符串最后的换行符 input[strlen(input) - 1] = '\0'; // 解析用户输入 command1 = strtok(input, " "); args1[0] = command1; for (i = 1; i < 10 && (args1[i] = strtok(NULL, " ")) != NULL; i++); if ((command2 = strtok(NULL, " ")) != NULL && strcmp(command2, "|") == 0) { args1[i-1] = NULL; command2 = strtok(NULL, " "); args2[0] = command2; for (i = 1; i < 10 && (args2[i] = strtok(NULL, " ")) != NULL; i++); args2[i] = NULL; } else { args1[i] = NULL; args2[0] = NULL; } // 创建管道 pipe(pipefd); // 创建第一个子进程 pid_t pid1 = fork(); if (pid1 == 0) { // 将stdout重定向管道的写入端 dup2(pipefd[1], STDOUT_FILENO); close(pipefd[0]); close(pipefd[1]); // 执行第一个命令 execvp(command1, args1); perror("execvp"); exit(EXIT_FAILURE); } // 创建第二个子进程 pid_t pid2 = fork(); if (pid2 == 0) { // 将stdin重定向管道的读取端 dup2(pipefd[0], STDIN_FILENO); close(pipefd[0]); close(pipefd[1]); // 执行第二个命令 execvp(command2, args2); perror("execvp"); exit(EXIT_FAILURE); } // 关闭管道 close(pipefd[0]); close(pipefd[1]); // 等待子进程结束 waitpid(pid1, NULL, 0); waitpid(pid2, NULL, 0); } return 0; } ``` 当用户输入命令时,如果包含管道操作符“|”,则程序会创建管道并创建两个子进程分别执行两个命令,并将第一个进程的stdout重定向管道的写入端,将第二个进程的stdin重定向管道的读取端。否则,程序会创建一个子进程执行该命令。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值