(轉自:http://woshiyouyouchen.blog.163.com/blog/static/103998942009112194240398/)
你還再為怎么判斷一個程序是否結束而苦惱嗎?或許你在想可以用一個循環每隔幾秒來判斷一次這個進程是否結束,但是這樣會有問題,如果間隔時間太短就會很占CPU資源,間隔時間太長又不能及時知道程序已經結束,或許有人馬上就想到WaitForSingleObject函數,不錯Win32 API提供的這個函數能很好的解決這個問題,但是在使用WaitForSingleObject函數的過程中很容易忽略一些問題。小弟不才,在這里略說一二。
先看一下函數的原形
DWORD WaitForSingleObject(HANDLE hHandle, DWORD dwMilliseconds);
參數hHandle是同步對象的句柄。
參數dwMilliseconds是以毫秒為單位的超時間隔,如果該參數為0,那么函數就測試同步對象的狀態並立即返回,如果該參數為INFINITE,則超時間隔是無限的。
表12.1 WaitForSingleObject的返回值
WaitForSingleObject函數能使線程阻塞其自身執行的等待函數。只有在作為其參數的同步對象產生信號時才會返回。或在超過規定的等待時間后,不管有無信號,函數也都會返回。在等待函數未返回時,線程處於等待狀態,此時線程只消耗很少的CPU時間。
1、等待自己創建的進程
怎么使用WaitForSingleObject函數呢,假設我們現在啟動系統的記事本程序,然后等待記事本結束,代碼可以這樣寫:
CString strCommand="";
STARTUPINFO si = {sizeof(si)};
PROCESS_INFORMATION pi;
CHAR cWindowsDir[MAX_PATH];
CHAR cCommand[MAX_PATH];
GetWindowsDirectory(cWindowsDir,MAX_PATH); //獲取系統目錄
strCommand= (CString)cWindowsDir+" \\NotePad.exe";
strcpy(cCommand,strCommand); //得到記事本的程序路徑
BOOL bRet = CreateProcess(NULL,cCommand,NULL,NULL,FALSE,0,NULL,NULL,&si,&pi);//啟動記事本程序
if (bRet)
{
if (!WaitForSingleObject(pi.hProcess,INFINITE))// 等待記事本程序關閉
{
AfxMessageBox("記事本已經關閉!");
}
}
2、等待已經存在的進程
現在我們知道記事本什么時候關閉了,那么關閉的時候我們就可以做我們想做的事情。但是一般的程序都不是通過CreateProcess創建的,要獲取已經在運行狀態的程序怎么辦呢?沒錯,要想辦法得到運行程序的進程ID,這就要借助OpenProcess函數,OpenProcess函數用來打開一個已經存在的進程對象,原形如下:
HANDLE OpenProcess ( DWORD dwDesiredAccess,
BOOL bInheritHandle,
DWORD dwProcessId
)
dwDesiredAccess參數,進程的操作權限。
bInheritHandle 句柄是否繼承
dwProcessId進程ID
返回值:如果調用成功,函數返回一個該進程的句柄,如果失敗返回NULL。
怎么獲取進程ID呢,需要用到另外幾個函數CreateToolHelp32Snapshot、Process32First、Process32Next,這幾個函數都是在TLHELP32.H中,用來一起完成對系統進程的枚舉(就像任務管理器把進程都枚舉出來一樣)
HANDLE WINAPI CreateToolhelp32Snapshot(
DWORD dwFlags,
DWORD th32ProcessID
);
dwFlags參數是一個標志,一般使用TH32CS_SNAPPROCESS,
th32ProcessID參數是進程ID,如果第一個參數是TH32CS_SNAPPROCESS,這個參數就會被忽略。
返回值,如果調用成功,返回一個Snapshot的句柄,失敗返回INVALID_HANDLE_VALUE;
BOOL WINAPI Process32First(
HANDLE hSnapshot,
LPPROCESSENTRY32 lppe
);
hSnapshot參數需要賦值為通過CreateToolHelp32Snapshot的句柄
lppe參數是進程的入口信息,需要定義一個PROCESSENTRY32類型的變量來獲取這個值。
Process32Next和 Process32First函數的參數意義一致,這三個函數一起就可以不斷對進程進行枚舉,直到Process32Next的返回值為FALSE。
那么可以這樣實現等待已經在運行的進程對象
PROCESSENTRY32 pe32;
HANDLE hSnaphot;
HANDLE hApp;
DWORD dProcess;
hSnaphot = CreateToolHelp32Snaphot(TH32CS_SNAPPROCESS,0); //獲取進程快照
Process32First(hSnaphot, &pe32); //指向第一個進程
do
{
if (lstrcmpi(pe32.szExeFile,_T("NotePad.exe"))==0) //查找進程名稱為NotePad.exe(記事本的程序)
{
dProcess = pe32.th32ProcessID;
break;
}
}while(Process32Next(hSnaphot, &pe32)); // 不斷循環直到取不到進程
hApp= OpenProcess(
PROCESS_VM_OPERATION|SYNCHRONIZE,FALSE,dProcess);
//根據進程ID獲取程序的句柄
if (!WaitForSingleObject(hApp,INFINITE))// 等待記事本程序關閉
{
AfxMessageBox("記事本已經關閉!");
}
到現在,我們不但知道了怎么使用WaitForSingleObject的等待函數,還知道了GetWindowsDirectory、CreateProcess、OpenProcess、CreateToolHelp32Snapshot、Process32First、Process32Next等函數的用法,這些函數一起實現了我們想要的功能,而且還在其他方面起着重要的作用
WaitForSingleObject函數能使線程阻塞其自身執行的等待函數。只有在作為其參數的同步對象產生信號時才會返回。或在超過規定的等待時間后,不管有無信號,函數也都會返回。在等待函數未返回時,線程處於等待狀態,此時線程只消耗很少的CPU時間。
1、等待自己創建的進程
怎么使用WaitForSingleObject函數呢,假設我們現在啟動系統的記事本程序,然后等待記事本結束,代碼可以這樣寫:
CString strCommand="";
STARTUPINFO si = {sizeof(si)};
PROCESS_INFORMATION pi;
CHAR cWindowsDir[MAX_PATH];
CHAR cCommand[MAX_PATH];
GetWindowsDirectory(cWindowsDir,MAX_PATH); //獲取系統目錄
strCommand= (CString)cWindowsDir+" \\NotePad.exe";
strcpy(cCommand,strCommand); //得到記事本的程序路徑
BOOL bRet = CreateProcess(NULL,cCommand,NULL,NULL,FALSE,0,NULL,NULL,&si,&pi);//啟動記事本程序
if (bRet)
{
if (!WaitForSingleObject(pi.hProcess,INFINITE))// 等待記事本程序關閉
{
AfxMessageBox("記事本已經關閉!");
}
}
2、等待已經存在的進程
現在我們知道記事本什么時候關閉了,那么關閉的時候我們就可以做我們想做的事情。但是一般的程序都不是通過CreateProcess創建的,要獲取已經在運行狀態的程序怎么辦呢?沒錯,要想辦法得到運行程序的進程ID,這就要借助OpenProcess函數,OpenProcess函數用來打開一個已經存在的進程對象,原形如下:
HANDLE OpenProcess ( DWORD dwDesiredAccess,
BOOL bInheritHandle,
DWORD dwProcessId
)
dwDesiredAccess參數,進程的操作權限。
bInheritHandle 句柄是否繼承
dwProcessId進程ID
返回值:如果調用成功,函數返回一個該進程的句柄,如果失敗返回NULL。
怎么獲取進程ID呢,需要用到另外幾個函數CreateToolHelp32Snapshot、Process32First、Process32Next,這幾個函數都是在TLHELP32.H中,用來一起完成對系統進程的枚舉(就像任務管理器把進程都枚舉出來一樣)
HANDLE WINAPI CreateToolhelp32Snapshot(
DWORD dwFlags,
DWORD th32ProcessID
);
dwFlags參數是一個標志,一般使用TH32CS_SNAPPROCESS,
th32ProcessID參數是進程ID,如果第一個參數是TH32CS_SNAPPROCESS,這個參數就會被忽略。
返回值,如果調用成功,返回一個Snapshot的句柄,失敗返回INVALID_HANDLE_VALUE;
BOOL WINAPI Process32First(
HANDLE hSnapshot,
LPPROCESSENTRY32 lppe
);
hSnapshot參數需要賦值為通過CreateToolHelp32Snapshot的句柄
lppe參數是進程的入口信息,需要定義一個PROCESSENTRY32類型的變量來獲取這個值。
Process32Next和 Process32First函數的參數意義一致,這三個函數一起就可以不斷對進程進行枚舉,直到Process32Next的返回值為FALSE。
那么可以這樣實現等待已經在運行的進程對象
PROCESSENTRY32 pe32;
HANDLE hSnaphot;
HANDLE hApp;
DWORD dProcess;
hSnaphot = CreateToolHelp32Snaphot(TH32CS_SNAPPROCESS,0); //獲取進程快照
Process32First(hSnaphot, &pe32); //指向第一個進程
do
{
if (lstrcmpi(pe32.szExeFile,_T("NotePad.exe"))==0) //查找進程名稱為NotePad.exe(記事本的程序)
{
dProcess = pe32.th32ProcessID;
break;
}
}while(Process32Next(hSnaphot, &pe32)); // 不斷循環直到取不到進程
hApp= OpenProcess(
PROCESS_VM_OPERATION|SYNCHRONIZE,FALSE,dProcess);
//根據進程ID獲取程序的句柄
if (!WaitForSingleObject(hApp,INFINITE))// 等待記事本程序關閉
{
AfxMessageBox("記事本已經關閉!");
}
到現在,我們不但知道了怎么使用WaitForSingleObject的等待函數,還知道了GetWindowsDirectory、CreateProcess、OpenProcess、CreateToolHelp32Snapshot、Process32First、Process32Next等函數的用法,這些函數一起實現了我們想要的功能,而且還在其他方面起着重要的作用