等待進程結束


(轉自: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等函數的用法,這些函數一起實現了我們想要的功能,而且還在其他方面起着重要的作用


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM