MFC单例运行三种方式
1、查找窗口句柄是否存在
2、互斥
3、注册表
转自http://k594081130.blog.163.com/blog/static/21835901320141220052740/
要使应用程序只允许同时运行一个实例,可以这样做:在XXXApp::InitInstance() 函数里面开头部位添加代码。
代码方案1:(直接查找该程序窗口是否已经存在)
CWnd *pWndPrev, *pWndChild;
if(pWndPrev=CWnd::FindWindow("#32770","DMTOOL") ) //见说明1
{
if(pWndPrev->IsIconic())
{
pWndPrev->ShowWindow(SW_RESTORE);
pWndPrev->SetForegroundWindow();
}
else
{
pWndChild=pWndPrev->GetLastActivePopup();
pWndChild->SetForegroundWindow();
}
return FALSE;
}
代码方案2:(使用互斥变量来判断)
HANDLE hMutex=CreateMutex(NULL,TRUE,"DMTOOL");
if(hMutex)
{
if(ERROR_ALREADY_EXISTS==GetLastError())
{
AfxMessageBox("程序已在运行中!");
return FALSE;
}
}
这里有几点说明: 1、FindWindow函数。函数第一个参数是窗口的类名,第二个参数是窗口标题。 窗口类名如何获得?使用SPY++里的查找功能,找到你所要看的窗口, 窗口属性里面有一项就是类名,例如这里的“#32770”。 2、GetLastActivePopup 函数功能:该函数确定指定窗口中的哪一个弹出式窗口是最近活动的窗口。 函数原型:HWND GetLastActivePopup(HWND hWnd); 参数:hWnd(所有者窗口句柄)。 返回值:返回值标识了最近活动的弹出式窗口的句柄。如果满足下列任一条件, 则返回值与参数hWnd相同: 由hWnd指定的窗口是最近活动的:由hWnd指定的窗口不拥有任何弹出式窗口; 由hWnd指定的窗口不是顶层窗口或它属于其他窗口。 3、SetForegroundWindow() 函数功能:该函数将创建指定窗口的线程设置到前台,并且激活该窗口。 键盘输入转向该窗口,并为用户改各种可视的记号。系统给创建前台窗口的线程 分配的权限稍高于其他线程。 函数原型:BOOL SetForegroundWindow(HWND hWnd) 参数: hWnd(将被激活并被调入前台的窗口句柄)。 返回值:如果窗口设入了前台,返回值为非零;如果窗口未被设入前台,返回值为零。 备注:前台窗口是z序顶部的窗口,是用户的工作窗口。 在一个多任务优先抢占环境中,应让用户控制前台窗口。
代码方案三:(注册码)
#define UNIQE_NAME "{1AB792D6-EAF2-3267-9A84-9135681127A4}"
#define GIS_MSG "{D48CA993-4925-41cb-8F59-ABAAAFCEF797}"
const UINT ID_GIS_INSTANCE = ::RegisterWindowMessage(GIS_MSG); //注册消息
// 确保程序只有一个实例在运行
m_hOneInstance = ::CreateMutex(NULL, FALSE, UNIQE_NAME);
if (GetLastError() == ERROR_ALREADY_EXISTS )//如果存在前一个
{
//发送消息,激活实例
DWORD dwRecipients = BSM_APPLICATIONS;
::BroadcastSystemMessage(BSF_NOHANG, &dwRecipients,
ID_GIS_INSTANCE, // registered window message
0, 0); // user defined parameters
return FALSE;
}
ON_REGISTERED_MESSAGE(ID_GIS_INSTANCE, OnGISInstance)
LRESULT CMainFrame::OnGISInstance(WPARAM wParam, LPARAM lParam)
{
theApp.m_pMainWnd->ShowWindow(SW_SHOWMAXIMIZED);
theApp.m_pMainWnd->SetForegroundWindow();
::AfxMessageBox("本程序已经有一个实例在运行了。");
return 0;
}