北京機械工業學院研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環境下的控制臺輸出定向。
|