最近,一个公司项目要求防止程序多开,采用了几种方法,效果还行。 一、使用Mutex 1、原理 创建一个互斥体,并检查它是否已经有拥有者,如果有,表明互斥体已经建立(程序已经启动),否则表明程序未启动。 2、实现 (1)首先创建一个互斥体,CreateMutex 函数,第一个参数可以设置为 NULL,第二个参数必须设置为 false,第三个参数表示互斥体的名称,这个名称最好有一些特殊标识以防止与其他应用程序冲突,比如程序名+时间。 (2)使用GetLastError()函数判断错误信息是否为 ERROR_ALREADY_EXISTS,如果是,则表示程序已经启动。 示例代码如下: view plaincopy to clipboardprint? 1. HANDLE hObject = ::CreateMutex(NULL,FALSE, _T("Mutex20100731")); 2. if(GetLastError() == ERROR_ALREADY_EXISTS) 3. { 4. CloseHandle(hObject); 5. MessageBox(NULL, _T("应用程序已经在运行!"), _T("提示"), MB_ICONERROR|MB_OK); 6. return FALSE; 7. } 3、效果 这个是非常简单的应用程序多开检测,一般的程序多开器均能破解此限制。 二、使用窗口属性 1、原理 在程序启动时,枚举桌面所有窗口,并检查其属性列表中是否存在特殊的属性值,如果有则表明程序已经启动,否则程序未启动。 2、实现 (1)程序启动时首先枚举所有窗口查找是否存在特定属性值,使用EnumWindows 函数遍历所有窗口。此函数需要一个回调函数,对于每一个窗口,都会调用此函数,并把遍历到的窗口句柄(HWND)传递给该函数,该回调函数原型如下: BOOL CALLBACK EnumWndProc(HWND hwnd, LPARAM lParam); lParam 可由 EnumWindows 的第二个参数传递。 (2)在EnumWndProc 回调函数中,我们需要获取窗口的属性值,然后检查是否和我们预定的属性值相同,如果相同,则表示程序已经启动。 (3)如果没有找到,我们需要将此特殊属性值设置到本程序的主窗口。 示例代码如下: view plaincopy to clipboardprint? 1. CString g_propName = _T("Prop20100731"); 2. HANDLE g_hValue = (HANDLE)1; 3. 4. BOOL CALLBACK EnumWndProc(HWND hwnd, LPARAM lParam) 5. { 6. HANDLE h = GetProp(hwnd, g_propName); 7. if(h == g_hValue) 8. { 9. *(HWND*)lParam = hwnd; 10. return FALSE; 11. } 12. return TRUE; 13. } 14. 15. BOOL CXxxxDlg::OnInitDialog() 16. { 17....