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