控制台应用程序的定向输出

发表于:2007-07-14来源:作者:点击数: 标签:
北京机械工业学院研00级 冉林仓 在 Windows 编程中,并非每一个应用程序都需要一个图形用户界面(GUI),很多情况下,我们可以编写一个控制台应用程序,这样程序更小,加载更快,传输时间也短,同时也丝毫不牺牲程序应有的功能。这种程序特别适合那些在后台运行的程
北京机械工业学院研00级 冉林仓

Windows编程中,并非每一个应用程序都需要一个图形用户界面(GUI),很多情况下,我们可以编写一个控制台应用程序,这样程序更小,加载更快,传输时间也短,同时也丝毫不牺牲程序应有的功能。这种程序特别适合那些在后台运行的程序,比如压缩、杀毒、上传下载等等。如果我们的确需要在GUI执行这些程序,以完成某些比如类似于磁盘格式化的功能,我们可以在GUI程序中创建一个新的进程,调用这些已有的控制台应用程序,帮助完成这些功能。然而令人失望的是,我们每次加载这些控制台应用程序时,图形程序总会在加载的过程中产生一个不受欢迎的控制台窗口,从而使我们图形用户界面显得不伦不类,当用户看到这个界面时,尤其看到我们加载的是别人编写的或者是操作系统提供的控制台应用程序,就会对我们产品的可信度表示怀疑,甚至大打折扣。因此我们必须竭力屏蔽这个窗口不让它显示出来,同时我们还需要把程序运行的结果定向到一个文本文件中,控制台程序的输入部分工作可以由交给GUI来完成。就像Visual C++编译一个程序一样,由MsDev.exe(GUI程序)负责加载编译器cl.exe(控制台程序)进行后台编译,然后把编译的结果定向到一个文件,并把编译结果输出到前台图形界面的一个窗口中,而用户在编译的过程中根本不会察觉这个过程,
C++为应用程序加载提供了多个函数,比如_spawnlp、ShellExecute、system、_exec等函数,这些函数除了system之外,都无法实现控制台程序的输出定向,而system函数的缺点是会导致一个控制台窗口出现,如果计算机配置是一个全屏命令提示行模式,它就会把你的GUI程序直接切换到全屏控制台窗口,显然这是一个很不体面的解决方案
_spawnlp( _P_WAIT,"netstat","-e","-s","-n","r","a","-p","ip",NULL);
::ShellExecute(NULL,NULL,"Ping.exe","168.192.0.1 >1.txt",NULL,SW_SHOWNORMAL);
system("Format a:/q >NULL");
_execlp("expand.exe","Source.cab","-f:m*.dll",c:\winnt\sytem32",NULL );
能够成功实现控制台应用程序输出定向的方法是调用CreateProcess函数。通过这个函数我们可以实现创建一个进程,能够隐藏控制台窗口,并把控制台窗口的输出结果定向输出到一个文本文件。
在Windows 2000环境下,CreateProcess函数提供了一个名叫CREATE_NO_WINDOW的标志,这个标志能够成功阻止控制台窗口出现,然而在Windows 98环境下,这个标志不被支持。为了实现两种环境下隐藏控制台窗口,我们可以通过设置STARTINFO结构成员并把它传递给CreateProcess函数来达到这个目的。
下面是程序实现部分的界面和部分代码:

UpdateData();
BYTE b1,b2,b3,b4;
if(m_IPAddressCtrl.GetAddress(b1,b2,b3,b4)<4){ file://获得IP地址的内容,不能空缺
m_IPAddressCtrl.SetFocus ();
return;
}
char cmdLine[MAX_PATH];
wsprintf(cmdLine,"Ping.exe %d.%d.%d.%d",b1,b2,b3,b4);
SECURITY_ATTRIBUTES sa={sizeof(sa),NULL,TRUE};
SECURITY_ATTRIBUTES *psa=NULL;
DWORD dwShareMode=FILE_SHARE_READ|FILE_SHARE_WRITE;
OSVERSIONINFO osVersion={0};
osVersion.dwOSVersionInfoSize =sizeof(osVersion);
if(GetVersionEx(&osVersion)){
if(osVersion.dwPlatformId ==VER_PLATFORM_WIN32_NT){
psa=&sa;
dwShareMode|=FILE_SHARE_DELETE;
}
}
file://根据版本设置共享模式和安全属性
HANDLE hConsoleRedirect=CreateFile(
"c:\\NetStatus.txt",
GENERIC_WRITE,
dwShareMode,
psa,
OPEN_ALWAYS,
FILE_ATTRIBUTE_NORMAL,
NULL);
ASSERT(hConsoleRedirect!=INVALID_HANDLE_VALUE);
STARTUPINFO s={sizeof(s)};
s.dwFlags =STARTF_USESHOWWINDOW|STARTF_USESTDHANDLES;
file://使用标准柄和显示窗口
s.hStdOutput =hConsoleRedirect;//将文件作为标准输出句柄
s.wShowWindow =SW_HIDE;//隐藏控制台窗口
PROCESS_INFORMATION pi={0};
if(CreateProcess(NULL,cmdLine,NULL,NULL,TRUE,NULL,NULL,NULL,&s,&pi)){
file://创建进程,执行Ping程序,测试网络是否连通
WaitForSingleObject(pi.hProcess ,INFINITE);
file://等待进程执行完毕
CloseHandle(pi.hProcess );
CloseHandle(pi.hThread );
file://关闭进程和主线程句柄
}
CloseHandle(hConsoleRedirect);
file://关闭控制台定向输出文件句柄
CFile myFile("c:\\NetStatus.txt",CFile::modeRead );
ASSERT (myFile.m_hFile!=NULL);  
char * pszNetStatus=new char[myFile.GetLength ()+1];
ZeroMemory(pszNetStatus,myFile.GetLength ()+1);
myFile.Read (pszNetStatus,myFile.GetLength ());
myFile.Close ();
file://打开文件,把它读到一个字符缓冲区
DeleteFile("c:\\NetStatus.txt");
file://删除临时文件
m_EditNetStatus.SetWindowText (pszNetStatus);
file://把控制台程序输出信息写到编辑框中
delete pszNetStatus;
本程序在Windows XP 环境下 用Microsoft Visual Studio.Net Beta 2调试通过,由于本程序没有使用visual c++ .net任何新的特性,利用上述代码,你完全可以用Visual C++ 6实现Windows2000 和Windows98环境下的控制台输出定向。 

原文转自:http://www.ltesting.net