C++ 判斷進程是否存在


原文:http://blog.csdn.net/u010803748/article/details/53927977?locationNum=2&fps=1


一、判斷指定程序名的進程是否存在

BOOL EnumWindows( WNDENUMPROC lpEnumFunc, // pointer to callback function LPARAM lParam // application-defined value);

The EnumWindows function enumerates all top-level windows on the screen by passing the handle to each window, in turn, to an application-defined callback function. EnumWindows continues until the last top-level window is enumerated or the callback function returns FALSE.

BOOL CALLBACK IpEnumFunc(HWND hwnd,LPARAM lParam) 
{ 
char wndName[100]; 
::GetWindowText(hwnd,wndName,sizeof(wndName)); 
if(wndName!="") 
{ 
if(strcmp(wndName,name1)==0) 
{ 
WndHnd=hwnd; 
flag=1; 

} 
} 
return 1; 
} 


二、判斷指定進程名的進程是否存在

DWORD GetProcessidFromName(LPCTSTR name) 
{ 
PROCESSENTRY32 pe; 
DWORD id=0; 
HANDLE hSnapshot=CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0); 
pe.dwSize=sizeof(PROCESSENTRY32); 
if(!Process32First(hSnapshot,&pe)) 
return 0; 
while(1) 
{ 
pe.dwSize=sizeof(PROCESSENTRY32); 
if(Process32Next(hSnapshot,&pe)==FALSE) 
break; 
if(strcmp(pe.szExeFile,name)==0) 
{ 
id=pe.th32ProcessID; 

break; 
} 


} 
CloseHandle(hSnapshot); 
return id; 
} 

如果返回值不為零,則存在,否則不存在。

三、VC判斷程序調用的外部進程是否結束

PROCESS_INFORMATION pi; 
STARTUPINFO si; 
memset(&si,0,sizeof(si)); 
si.cb=sizeof(si); 
si.wShowWindow=SW_HIDE; 
si.dwFlags=STARTF_USESHOWWINDOW; 
bool fRet=CreateProcess(NULL,str.GetBuffer(str.GetLength()),NULL,FALSE,NULL,NORMAL_PRIORITY_CLASS | CREATE_NO_WINDOW,NULL,NULL,&si,&pi); 

///判斷 

DWORD ExitCode; 

ExitCode=STILL_ACTIVE; 
while(ExitCode==STILL_ACTIVE) 
{ 
GetExitCodeProcess(pi.hProcess,&ExitCode); 
} 


四、VC判斷進程是否存在?比如我想知道記事本是否運行,要用到哪些函數?

enProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE,aProcesses[i]); 
// 取得特定PID的進程名 
if (hProcess ) 
{ 
if ( EnumProcessModules(hProcess,&hMod,sizeof(hMod), &cbNeeded)) 
{ 
GetModuleBaseName( hProcess, hMod,szProcessName,sizeof(szProcessName)); 
//將取得的進程名與輸入的進程名比較,如相同則返回進程PID 
if(!stricmp(szProcessName, InputProcessName)) 
{ 
CloseHandle(hProcess); 
return aProcesses[i]; 
} 
} 
}//end of if (hProcess) 
}//end of for 
//沒有找到相應的進程名,返回0 
CloseHandle(hProcess); 
return 0; 
} 

也可以枚舉得到所有進程的應用程序名,然后和知道應用程序名比較判斷。

五、實現程序只運行一次的方法

實現程序只運行一次的方法很多,但是原理都是一樣的,就是運行第一次的時候設置一個標記,每次運行的時候檢查該標記,如果有就說明已經運行了。

具體實現:

1、在程序初始化的時候 (InitInstance()) 枚舉所有的窗口,查找本程序的實例是否存在
2、在主窗口初始化的時候在本窗口的屬性列表中添加一個標記,以便程序查找.


部分關鍵代碼 :

1、在App的InitInstance()中枚舉所有窗口,查找本程序實例

HWND oldHWnd = NULL; 
EnumWindows(EnumWndProc,(LPARAM)&oldHWnd); //枚舉所有運行的窗口 
if(oldHWnd != NULL) 
{ 
AfxMessageBox( "本程序已經在運行了 "); 
::ShowWindow(oldHWnd,SW_SHOWNORMAL); //激活找到的前一個程序 
::SetForegroundWindow(oldHWnd); //把它設為前景窗口 
return false; //退出本次運行 
} 

  

2、添加EnumWndProc窗口過程函數://添加的標識只運行一次的屬性名

CString g_szPropName = "Your Prop Name "; //自己定義一個屬性名 
HANDLE g_hValue = (HANDLE)1; //自己定義一個屬性值 

BOOL CALLBACK EnumWndProc(HWND hwnd,LPARAM lParam) 
{ 
HANDLE h = GetProp(hwnd,g_szPropName); 
if( h == g_hValue) 
{ 
*(HWND*)lParam = hwnd; 
return false; 
} 
return true; 
} 

  

3、在主窗口的 OnInitDialog()中添加屬性 //設置窗口屬性
SetProp(m_hWnd,g_szPropName,g_hValue);

再次啟動時,先檢查當前存在的所有窗口,如果有標題相同的,則把先前運行的窗口當成當前窗口
我的程序如下:

HWND hWnd_Exist; 
hWnd_Exist=::GetDesktopWindow(); 
hWnd_Exist=::GetWindow(hWnd_Exist,GW_CHILD); 
for(;;) 
{ 
if(hWnd_Exist==NULL) 
{ 
break; 
} 
char s[256]; 
memset(s,0,256); 
::SendMessage(hWnd_Exist,WM_GETTEXT,255,(LONG)s); 
if(strstr(s, "****** ")!=NULL) 
break; 
hWnd_Exist=::GetWindow(hWnd_Exist,GW_HWNDNEXT); 
} 

if(hWnd_Exist != NULL) 
{ 
::ShowWindow(hWnd_Exist,SW_SHOWNORMAL); 
::SetForegroundWindow(hWnd_Exist); 
exit(0); 
} 

  

聲明一個全局 CMutex 變量:
CMutex mutexApp(FALSE, _T( "VPOS2000Server ")); //用此互斥量阻止多個實例

在你的 CWinApp 類的重載函數: InitInstance 中加入如下代碼:

if (!mutexApp.Lock(1)) 
return FALSE; 
::CreateMutex(NULL, TRUE, m_pszExeName); 
if(ERROR_ALREADY_EXISTS == GetLastError()) 
{ 
CWnd* pPrevHwnd = CWnd::GetDesktopWindow()-> GetWindow(GW_CHILD); 
while(pPrevHwnd) 
{ 
if(::GetProp(pPrevHwnd-> GetSafeHwnd(), m_pszExeName)) 
{ 
if(pPrevHwnd-> IsIconic()) 
{ 
pPrevHwnd-> ShowWindow(SW_RESTORE); 
} 

pPrevHwnd-> SetForegroundWindow(); 
pPrevHwnd-> GetLastActivePopup()-> SetForegroundWindow(); 
return FALSE; 
} 
pPrevHwnd = pPrevHwnd-> GetWindow(GW_HWNDNEXT); 
} 
TRACE( "Could not fond frevious instance main window ! "); 
return FALSE; 
} 

  

創建一個全局的互斥量,每次啟動時檢查是否存在。

BOOL CRTDBApp::OnlyOneInstance() 
{ 
if(::CreateMutex(NULL, TRUE, "onlyone ") == NULL ) 
{ 
TRACE0( "CreateMutex error. "); 
return FALSE; 
}; 
if( ::GetLastError() == ERROR_ALREADY_EXISTS) { 

CWnd* pPrevWnd = CWnd::FindWindow(NULL, "onlyonehwnd "); 
if(pPrevWnd) 
{ 
if( pPrevWnd-> IsIconic()) 
pPrevWnd-> ShowWindow(SW_RESTORE); 

pPrevWnd-> SetForegroundWindow(); 

pPrevWnd-> GetLastActivePopup()-> SetForegroundWindow(); 
return FALSE; 
} 

}; 

return TRUE; 
}

  




免責聲明!

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



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