北京機械工業(yè)學院研00級 冉林倉
在Windows編程中,并非每一個應用程序都需要一個圖形用戶界面(GUI),很多情況下,我們可以編寫一個控制臺應用程序,這樣程序更小,加載更快,傳輸時間也短,同時也絲毫不犧牲程序應有的功能。這種程序特別適合那些在后臺運行的程序,比如壓縮、殺毒、上傳下載等等。如果我們的確需要在GUI執(zhí)行這些程序,以完成某些比如類似于磁盤格式化的功能,我們可以在GUI程序中創(chuàng)建一個新的進程,調(diào)用這些已有的控制臺應用程序,幫助完成這些功能。然而令人失望的是,我們每次加載這些控制臺應用程序時,圖形程序總會在加載的過程中產(chǎn)生一個不受歡迎的控制臺窗口,從而使我們圖形用戶界面顯得不倫不類,當用戶看到這個界面時,尤其看到我們加載的是別人編寫的或者是操作系統(tǒng)提供的控制臺應用程序,就會對我們產(chǎn)品的可信度表示懷疑,甚至大打折扣。因此我們必須竭力屏蔽這個窗口不讓它顯示出來,同時我們還需要把程序運行的結(jié)果定向到一個文本文件中,控制臺程序的輸入部分工作可以由交給GUI來完成。就像Visual C++編譯一個程序一樣,由MsDev.exe(GUI程序)負責加載編譯器cl.exe(控制臺程序)進行后臺編譯,然后把編譯的結(jié)果定向到一個文件,并把編譯結(jié)果輸出到前臺圖形界面的一個窗口中,而用戶在編譯的過程中根本不會察覺這個過程, C++為應用程序加載提供了多個函數(shù),比如_spawnlp、ShellExecute、system、_exec等函數(shù),這些函數(shù)除了system之外,都無法實現(xiàn)控制臺程序的輸出定向,而system函數(shù)的缺點是會導致一個控制臺窗口出現(xiàn),如果計算機配置是一個全屏命令提示行模式,它就會把你的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 ); 能夠成功實現(xiàn)控制臺應用程序輸出定向的方法是調(diào)用CreateProcess函數(shù)。通過這個函數(shù)我們可以實現(xiàn)創(chuàng)建一個進程,能夠隱藏控制臺窗口,并把控制臺窗口的輸出結(jié)果定向輸出到一個文本文件。 在Windows 2000環(huán)境下,CreateProcess函數(shù)提供了一個名叫CREATE_NO_WINDOW的標志,這個標志能夠成功阻止控制臺窗口出現(xiàn),然而在Windows 98環(huán)境下,這個標志不被支持。為了實現(xiàn)兩種環(huán)境下隱藏控制臺窗口,我們可以通過設(shè)置STARTINFO結(jié)構(gòu)成員并把它傳遞給CreateProcess函數(shù)來達到這個目的。 下面是程序?qū)崿F(xiàn)部分的界面和部分代碼: UpdateData(); BYTE b1,b2,b3,b4; if(m_IPAddressCtrl.GetAddress(b1,b2,b3,b4)<4){ file://獲得IP地址的內(nèi)容,不能空缺 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://根據(jù)版本設(shè)置共享模式和安全屬性 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://創(chuàng)建進程,執(zhí)行Ping程序,測試網(wǎng)絡(luò)是否連通 WaitForSingleObject(pi.hProcess ,INFINITE); file://等待進程執(zhí)行完畢 CloseHandle(pi.hProcess ); CloseHandle(pi.hThread ); file://關(guān)閉進程和主線程句柄 } CloseHandle(hConsoleRedirect); file://關(guān)閉控制臺定向輸出文件句柄 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://打開文件,把它讀到一個字符緩沖區(qū) DeleteFile("c:\\NetStatus.txt"); file://刪除臨時文件 m_EditNetStatus.SetWindowText (pszNetStatus); file://把控制臺程序輸出信息寫到編輯框中 delete pszNetStatus; 本程序在Windows XP 環(huán)境下 用Microsoft Visual Studio.Net Beta 2調(diào)試通過,由于本程序沒有使用visual c++ .net任何新的特性,利用上述代碼,你完全可以用Visual C++ 6實現(xiàn)Windows2000 和Windows98環(huán)境下的控制臺輸出定向。
|