北京机械工业学院研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环境下的控制台输出定向。