進程只是提供了一段地址空間和內核對象,其運行時通過其他地址空間內的主線程來體現的。當主線程的進入點函數返回時,進程也就隨之而技術。這種進程的種植方式是進程的正常退出。進程中的所有縣城資源都能夠得到正確的清除。除了這種進程的正常退出方式之外,優勢還需要在程序中通過代碼來強制結束本進程或其他進程的運行。
ExitProcess
void ExitProcess(UINT uExitCode);
其參數uExitCode為進城設置了退出代碼。該函數具有強制性,在執行完畢后進程即被結束,因此位於其后的任何代碼將不能被執行。雖然 ExitProcess()函數可以再結束進程同時通知與其關聯的動態鏈接庫,但是由於他的這種強制性,使得ExitProcess()函數在使用上將存有安全隱患。例如,如果最親愛程序調用ExitProcess()函數之前曾用new操作,申請一段空間,那么敬愛那個會由於ExitProcess() 函數的強制性而無法通過delete操作符將其釋放,從而造成內存泄露。
有鑒於ExitProcess()函數的強制性和安全性,在使用時一定要引起注意。
Terminateprocess()
ExitProcess 只能強制本進程的推出,如果要在一個進程中強制結束其他的進程就需要用TerminateProcess()來實現,與ExitProcess()不同,TerminateProcess()函數執行后,被終止的進程不會得到任何關於程序退出的通知。也就是說,被終止的進程是無法再結束運行前進程推出前的收尾工作的。所以,通常只有在其他任何地方都無法迫使進程退出時才會考慮使用TerminateProcess()去強制結束進程。
BOOL TerminateProcess(HANDLE hProcess, UINT uExitCode);
參數hProcess和uExitCode分別為進城句柄和退出代碼。如果被結束的是本進程,可以通過GetCurrentProcess()獲取到句柄。 TerminateProcess()是異步執行的,在調用后返回並不能確定被終止進程是否已經真的退出,如果調用TerminateProcess() 的進程對此細節關心,可以通過WaitForSingleObject()來等待進程的真正結束。
在VC中如何結束系統正在運行的其他進程(該進程必須有窗口界面),其實很簡單,按照如下步驟進程:
1)取得進程的句柄(利用FindWindow函數得到);
2)獲取進程ID號(用GetWindowThreadProcessId函數獲取);
3)打開進程,OpenProcess函數中的第一個參數設為PROCESS_TERMINATE,就可以獲取處理該進程的句柄;
4)利用TerminateProcess函數結束進程,將該函數的第二個參數設為4.
代碼如下:
//結束進程
int CStaticFunc::KillProcess(LPCSTR pszClassName, LPCSTR
pszWindowTitle)
{
HANDLE hProcessHandle;
ULONG nProcessID;
HWND TheWindow;
TheWindow = ::FindWindow( NULL, pszWindowTitle );
::GetWindowThreadProcessId( TheWindow, &nProcessID );
hProcessHandle = ::OpenProcess( PROCESS_TERMINATE, FALSE,
nProcessID );
return ::TerminateProcess( hProcessHandle, 4 );
}
而啟動進程則只需要CreateProcess函數就可完成,需要注意的是這個函數的幾個輸入參數,第一個參數是
//啟動新進程
int CStaticFunc::CreateNewProcess(LPCSTR pszExeName)
{
PROCESS_INFORMATION piProcInfoGPS;
STARTUPINFO siStartupInfo;
SECURITY_ATTRIBUTES saProcess, saThread;
ZeroMemory( &siStartupInfo, sizeof(siStartupInfo) );
siStartupInfo.cb = sizeof(siStartupInfo);
saProcess.nLength = sizeof(saProcess);
saProcess.lpSecurityDescriptor = NULL;
saProcess.bInheritHandle = true;
saThread.nLength = sizeof(saThread);
saThread.lpSecurityDescriptor = NULL;
saThread.bInheritHandle = true;
return ::CreateProcess( NULL, (LPTSTR)pszExeName, &saProcess,
& saThread, false,
Create_DEFAULT_ERROR_MODE, NULL, NULL,
& siStartupInfo, &piProcInfoGPS );
}