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