有独占要求的程序 防止程序启动多次的方法

发布时间:2016-12-9 19:53:47 编辑:www.fx114.net 分享查询网我要评论
本篇文章主要介绍了"有独占要求的程序 防止程序启动多次的方法",主要涉及到有独占要求的程序 防止程序启动多次的方法方面的内容,对于有独占要求的程序 防止程序启动多次的方法感兴趣的同学可以参考一下。

http://blog.csdn.net/RobertSoft/archive/2010/08/01/5778149.aspx 最近,一个公司项目要求防止程序多开,采用了几种方法,效果还行。 一、使用Mutex       1、原理        创建一个互斥体,并检查它是否已经有拥有者,如果有,表明互斥体已经建立(程序已经启动),否则表明程序未启动。        2、实现        (1)首先创建一个互斥体,CreateMutex函数,第一个参数可以设置为NULL,第二个参数必须设置为false,第三个参数表示互斥体的名称,这个名称最好有一些特殊标识以防止与其他应用程序冲突,比如程序名+时间。        (2)使用GetLastError()函数判断错误信息是否为ERROR_ALREADY_EXISTS,如果是,则表示程序已经启动。        示例代码如下: HANDLE hObject = ::CreateMutex(NULL,FALSE, _T("Mutex20100731"));  if(GetLastError() == ERROR_ALREADY_EXISTS)      {          CloseHandle(hObject);          MessageBox(NULL, _T("应用程序已经在运行!"), _T("提示"), MB_ICONERROR|MB_OK);          return FALSE;      }         3、效果        这个是非常简单的应用程序多开检测,一般的程序多开器均能破解此限制。 二、使用窗口属性       1、原理       在程序启动时,枚举桌面所有窗口,并检查其属性列表中是否存在特殊的属性值,如果有则表明程序已经启动,否则程序未启动。       2、实现       (1)程序启动时首先枚举所有窗口查找是否存在特定属性值,使用EnumWindows函数遍历所有窗口。此函数需要一个回调函数,对于每一个窗口,都会调用此函数,并把遍历到的窗口句柄(HWND)传递给该函数,该回调函数原型如下:       BOOL CALLBACK EnumWndProc(HWND hwnd, LPARAM lParam); lParam可由EnumWindows的第二个参数传递。       (2)在EnumWndProc回调函数中,我们需要获取窗口的属性值,然后检查是否和我们预定的属性值相同,如果相同,则表示程序已经启动。       (3)如果没有找到,我们需要将此特殊属性值设置到本程序的主窗口。       示例代码如下:     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("应用程序已经在运行!"), _T("提示"), MB_ICONERROR|MB_OK);             DestroyWindow();             return FALSE;      }      SetProp(m_hWnd, g_propName, g_hValue);  }        3、效果       没有做过多的测试,手头有两个多开器均不能多开。 三、使用公共文件       1、原理       程序启动时,在一个公共目录(比如C:\或者Temp目录)中创建一个公共文件,并将此文件设置为不共享读写。第二个程序启动时,也打开此文件,如果打开成功,则表示程序未启动过,否则表示程序已经启动。       2、实现       此方法实现较为简单,不做详细说明了,请自行查阅CFile等相关文件操作。       3、效果       多开器肯定是不能够多开了,但是可以手动设置多开。比如:设定文件访问权限,不允许此程序在公共目录创建文件等。应对方法就是,如果不能创建文件则程序不允许运行。 四、mac地址验证       1、原理       必须是网络应用程序,如果单机运行,此方法无效。       登陆服务器时,获取本机mac地址,发送至服务器端,服务端进行mac地址验证,如果mac地址重复登陆,则不允许同服务器进行消息传递。       2、实现       客户端主要是mac地址获取,这个问题我至今没有找到太好的解决方案,效果较好的方法是读取注册表获取。       首先使用GetAdaptersInfo函数获取所有网卡信息,然后,对于每一个网卡信息查找注册表HKEY_LOCAL_MACHINE\\System\\CurrentControlSet\\Control\\Network\\{4D36E972-E325-11CE-BFC1-08002BE10318}\\网卡名称\\Connection位置,如果MediaSubType的值为0x01并且PnpInstanceID中含有PCI字串则表示是物理网卡。      3、效果      差强人意,多开器倒是不能用了,但是可以使用超级兔子等软件修改mac地址实现。 五、查看网络连接      1、原理      必须是网络应用程序,如果是单机运行,则此方法无效。      获取本机所有网络连接,检查是否有连接到服务器IP和端口号的连接,如果有,表示程序已经启动,否则程序未启动。     2、实现     使用GetTcpTable获取TCP连接,使用GetUdpTable获取UDP连接。需要注意的是,其获取的ip和端口号都是一个DWORD值,并且高低位相反。IP地址可以通过inet_addr函数将字符串形式的IP地址(如"127.0.0.1")转换为DWORD型的,端口号可以使用以下公式转换: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];//服务器IP     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("应用程序已经在运行!"), _T("提示"), MB_ICONERROR|MB_OK);              return FALSE;          }      }  }  delete []pTcpTable;      3、效果     多开器肯定不能用,但有其他方式导致GetTcpTable函数失败(比如挂系统钩子等)。     总结了以上几种方法,具体哪种适合,还需要根据实际应用情况来判断,也可以几种方法混合使用,加强效果。 1.CreateMutex() 该函数介绍在百度百科比较详细: http://baike.baidu.com/link?url=b8XTzmD837H6WtmGu8tFn9Rvdh5v935KmpayeAngUMI-Xc4Mpub2f5kdb8Ua3kR7JPXuBb91Of6w913JozurzK 在工程文件中, WinMain函数里加上以下代码(此代码在BCB6.0下运行): HANDLE hMutex = CreateMutex(NULL, false, "Process");          if (GetLastError() == ERROR_ALREADY_EXISTS)          {             CloseHandle(hMutex);             MessageBox(Application->Handle, "程序已经在运行中,不能重复启动!", "提示", MB_OK +MB_ICONWARNING);             Application->Terminate();             return 0;          }          Application->CreateForm(__classid(TForm1), &Form1); 主要使用到CreateMutex()函数和GetLastError()以及一个常量ERROR_ALREADY_EXISTS. 2.如果程序有窗体的话, 还可以使用FindWindow().  void *handle = FindWindow(NULL, WindowName.c_str());  if (handle!=NULL)     return 0; 

上一篇:数据结构——循环队列
下一篇:centos安装和卸载软件

相关文章

相关评论