最近,一個公司項目要求防止程序多開,采用了幾種方法,效果還行。
一、使用Mutex
1、原理
創(chuàng)建一個互斥體,并檢查它是否已經(jīng)有擁有者,如果有,表明互斥體已經(jīng)建立(程序已經(jīng)啟動),否則表明程序未啟動。
2、實現(xiàn)
(1)首先創(chuàng)建一個互斥體,CreateMutex函數(shù),第一個參數(shù)可以設(shè)置為NULL,第二個參數(shù)必須設(shè)置為false,第三個參數(shù)表示互斥體的名稱,這個名稱最好有一些特殊標(biāo)識以防止與其他應(yīng)用程序沖突,比如程序名+時間。
(2)使用GetLastError()函數(shù)判斷錯誤信息是否為ERROR_ALREADY_EXISTS,如果是,則表示程序已經(jīng)啟動。
示例代碼如下:
- HANDLE hObject = ::CreateMutex(NULL,FALSE, _T("Mutex20100731"));
- if(GetLastError() == ERROR_ALREADY_EXISTS)
- {
- CloseHandle(hObject);
- MessageBox(NULL, _T("應(yīng)用程序已經(jīng)在運行!"), _T("提示"), MB_ICONERROR|MB_OK);
- return FALSE;
- }
3、效果
這個是非常簡單的應(yīng)用程序多開檢測,一般的程序多開器均能破解此限制。
二、使用窗口屬性
1、原理
在程序啟動時,枚舉桌面所有窗口,并檢查其屬性列表中是否存在特殊的屬性值,如果有則表明程序已經(jīng)啟動,否則程序未啟動。
2、實現(xiàn)
(1)程序啟動時首先枚舉所有窗口查找是否存在特定屬性值,使用EnumWindows函數(shù)遍歷所有窗口。此函數(shù)需要一個回調(diào)函數(shù),對于每一個窗口,都會調(diào)用此函數(shù),并把遍歷到的窗口句柄(HWND)傳遞給該函數(shù),該回調(diào)函數(shù)原型如下:
BOOL CALLBACK EnumWndProc(HWND hwnd, LPARAM lParam);
lParam可由EnumWindows的第二個參數(shù)傳遞。
(2)在EnumWndProc回調(diào)函數(shù)中,我們需要獲取窗口的屬性值,然后檢查是否和我們預(yù)定的屬性值相同,如果相同,則表示程序已經(jīng)啟動。
(3)如果沒有找到,我們需要將此特殊屬性值設(shè)置到本程序的主窗口。
示例代碼如下:
- CString g_propName = _T("Prop20100731");
- HANDLE g_hValue = (HANDLE)1;
-
- BOOL CALLBACK EnumWndProc(HWND hwnd, LPARAM lParam)
- {
- HANDLE h = GetProp(hwnd, g_propName);
- if(h == g_hValue)
- {
- *(HWND*)lParam = hwnd;
- return FALSE;
- }
- return TRUE;
- }
-
- BOOL CXxxxDlg::OnInitDialog()
- {
- CDialog::OnInitDialog();
-
-
- HWND hOldWnd = NULL;
- EnumWindows(EnumWndProc, (LPARAM)&hOldWnd);
- if(IsWindow(hOldWnd))
- {
- MessageBox(NULL, _T("應(yīng)用程序已經(jīng)在運行!"), _T("提示"), MB_ICONERROR|MB_OK);
- DestroyWindow();
- return FALSE;
- }
- SetProp(m_hWnd, g_propName, g_hValue);
- }
3、效果
沒有做過多的測試,手頭有兩個多開器均不能多開。
三、使用公共文件
1、原理
程序啟動時,在一個公共目錄(比如C:\或者Temp目錄)中創(chuàng)建一個公共文件,并將此文件設(shè)置為不共享讀寫。第二個程序啟動時,也打開此文件,如果打開成功,則表示程序未啟動過,否則表示程序已經(jīng)啟動。
2、實現(xiàn)
此方法實現(xiàn)較為簡單,不做詳細(xì)說明了,請自行查閱CFile等相關(guān)文件操作。
3、效果
多開器肯定是不能夠多開了,但是可以手動設(shè)置多開。比如:設(shè)定文件訪問權(quán)限,不允許此程序在公共目錄創(chuàng)建文件等。應(yīng)對方法就是,如果不能創(chuàng)建文件則程序不允許運行。
四、mac地址驗證
1、原理
必須是網(wǎng)絡(luò)應(yīng)用程序,如果單機運行,此方法無效。
登陸服務(wù)器時,獲取本機mac地址,發(fā)送至服務(wù)器端,服務(wù)端進(jìn)行mac地址驗證,如果mac地址重復(fù)登陸,則不允許同服務(wù)器進(jìn)行消息傳遞。
2、實現(xiàn)
客戶端主要是mac地址獲取,這個問題我至今沒有找到太好的解決方案,效果較好的方法是讀取注冊表獲取。
首先使用GetAdaptersInfo函數(shù)獲取所有網(wǎng)卡信息,然后,對于每一個網(wǎng)卡信息查找注冊表HKEY_LOCAL_MACHINE\\System\\CurrentControlSet\\Control\\Network\\{4D36E972-E325-11CE-BFC1-08002BE10318}\\網(wǎng)卡名稱\\Connection位置,如果MediaSubType的值為0x01并且PnpInstanceID中含有PCI字串則表示是物理網(wǎng)卡。
3、效果
差強人意,多開器倒是不能用了,但是可以使用超級兔子等軟件修改mac地址實現(xiàn)。
五、查看網(wǎng)絡(luò)連接
1、原理
必須是網(wǎng)絡(luò)應(yīng)用程序,如果是單機運行,則此方法無效。
獲取本機所有網(wǎng)絡(luò)連接,檢查是否有連接到服務(wù)器IP和端口號的連接,如果有,表示程序已經(jīng)啟動,否則程序未啟動。
2、實現(xiàn)
使用GetTcpTable獲取TCP連接,使用GetUdpTable獲取UDP連接。需要注意的是,其獲取的ip和端口號都是一個DWORD值,并且高低位相反。IP地址可以通過inet_addr函數(shù)將字符串形式的IP地址(如“127.0.0.1”)轉(zhuǎn)換為DWORD型的,端口號可以使用以下公式轉(zhuǎn)換:DWORD dwPort = ((nPort & 0xff) << 8) + ((nPort & 0xff00) >> 8);
示例代碼如下:
- PMIB_TCPTABLE pTcpTable = new MIB_TCPTABLE[1];
- DWORD dwSize = 0;
- if(GetTcpTable(pTcpTable, &dwSize, TRUE) == ERROR_INSUFFICIENT_BUFFER)
- {
- delete pTcpTable;
- pTcpTable = new MIB_TCPTABLE[dwSize / sizeof(MIB_TCPTABLE)];
- }
- if(GetTcpTable(pTcpTable, &dwSize, FALSE) == NO_ERROR)
- {
- char cServerAddr[100];
- int nPort;
- DWORD dwIP = inet_addr(cServerAddr);
- DWORD dwPort = ((nPort & 0xff) << 8) + ((nPort & 0xff00) >> 8);
- for (int i = 0; i < (int) pTcpTable->dwNumEntries; i++)
- {
- if(pTcpTable->table[i].dwRemoteAddr == dwIP
- && pTcpTable->table[i].dwRemotePort == dwPort)
- {
- MessageBox(gDataCenter.GetMainWnd(), _T("應(yīng)用程序已經(jīng)在運行!"), _T("提示"), MB_ICONERROR|MB_OK);
- return FALSE;
- }
- }
- }
- delete []pTcpTable;
3、效果
多開器肯定不能用,但有其他方式導(dǎo)致GetTcpTable函數(shù)失敗(比如掛系統(tǒng)鉤子等)。
總結(jié)了以上幾種方法,具體哪種適合,還需要根據(jù)實際應(yīng)用情況來判斷,也可以幾種方法混合使用,加強效果。