程序只运行一个实例

本篇是参考了互联网上的资料,感谢慷慨的分享。

在开发过程中会遇到只允许运行一个实例的情况。 今天对这个问题进行探讨。
  • 实现原理
大家都看过或者使用过类似只运行一个实例的程序,比如:QQ游戏、部分浏览器 等等! 让一个程序只运行一个实例的方法有多种,但是原理都类似,也就是在程序创建后,有窗口的程序在窗口创建前, 检查系统中是否已经设置了某些特定标志,是否创建了一些全局唯一的东西,或者让程序的多个实例都能看到的东西, 如果有则说明已经有一个实例在运行了,则当前程序通知用户如何如何,然后程序退出,当然方法有很多种,各有各的优缺点!
  • 方法一:
    创建互斥体 Mutex 法:
    但是单纯的使用互斥体的话不能取得已经创建的实例窗口局柄,因此无法激活已经启动的实例窗口;
HANDLE m_hMutex = ::CreateMutex(NULL, FALSE, _T("{6668BB0A-DE0C-499d-8520-79653FF9B2EB}"));
if ( GetLastError() == ERROR_ALREADY_EXISTS ){
    AfxMessageBox(_T("已经有一个实例正在运行中……"));
    CloseHandle(m_hMutex);
    m_hMutex = NULL;
    return FALSE;
}
//其他代码,比如对话框的创建及弹出等等
if (m_hMutex) {
    CloseHandle(m_hMutex);
    m_hMutex = NULL;
}
  • 方法二:
    通过 FindWindow 进行窗口的查找,若发现则说明已经运行过一个实例,并将其窗口激活:
HWND hWnd = ::FindWindow(_T("#32770"), _T("DlgTest"));
if (hWnd != NULL) {
    AfxMessageBox(_T("已经有一个实例正在运行中……"));
    ::ShowWindow(hWnd, SW_NORMAL);
    ::SetForegroundWindow(hWnd);
    return FALSE;
}

此种方法不是很好,如果窗口标题改变了或者每个窗口实例的标题不一样,就找不到了!

  • 方法三:
    设置窗口属性 SetProp + EnumWindows 法!
    1、加入全局变量的定义及枚举窗口函数:
TCHAR g_szPropName[] = _T("{12AA5160-5215-435b-AE3C-60C9E65615CE}");
HANDLE g_hValue = (HANDLE)9527;

BOOL CALLBACK EnumWndProc(HWND hwnd, LPARAM lParam)
{
    HANDLE hProp = GetProp(hwnd, g_szPropName);
    if(hProp == g_hValue) {
        *(HWND *)lParam = hwnd;
        return FALSE;
    }
    return TRUE;
}

2、窗口的枚举以及属性的添加:

//OnInitDialog() 中加入以下代码:
HWND hPreWnd = NULL;
::EnumWindows(EnumWndProc, (LPARAM)&hPreWnd);
if (hPreWnd != NULL) {
    AfxMessageBox(_T("已经有一个实例正在运行中……"));
    ::ShowWindow(hPreWnd, SW_NORMAL);
    ::SetForegroundWindow(hPreWnd);
    ExitProcess(0);
    return FALSE;
}
::SetProp(m_hWnd, g_szPropName, g_hValue);

3、窗口属性的删除:

OnDestroy() 函数中加入以下代码:
::RemoveProp(m_hWnd, g_szPropName);
  • 方法四
    使用全局共享变量的共享节法实现单实例运行;
    1、新建共享节:
#pragma data_seg("Shared")
HWND hPreWnd = NULL;
#pragma data_seg()
#pragma comment(linker, "/Section:Shared,RWS")

2、OnInitDialog() 函数中加入以下代码:

if (hPreWnd == NULL) {
    hPreWnd = m_hWnd;
} else {
    AfxMessageBox(_T("已经有一个实例正在运行中……"));
    ::ShowWindow(hPreWnd, SW_NORMAL);
    ::SetForegroundWindow(hPreWnd);
    ExitProcess(0);
    return FALSE;
}

===================================================
- 方法五:
互斥体+自定义广播系统消息法;
1、系统消息的注册:

#define REG_MSG (_T("{7510EF00-BADA-48de-A6CE-5FBC817616DD}"))
UINT WM_ACTIVE_MSG = ::RegisterWindowMessage(REG_MSG);

2、发现实例后,进行消息的广播:

InitInstance() 函数中添加如下代码:

HANDLE m_hMutex = ::CreateMutex(NULL, FALSE, _T("{6668BB0A-DE0C-499d-8520-79653FF9B2EB}"));
if ( GetLastError() == ERROR_ALREADY_EXISTS ){
    AfxMessageBox(_T("已经有一个实例正在运行中……"));
    CloseHandle(m_hMutex);
    m_hMutex = NULL;

    DWORD dwRecipients = BSM_APPLICATIONS;
    ::BroadcastSystemMessage(BSF_NOHANG, &dwRecipients, WM_ACTIVE_MSG, 0, 0);

    return FALSE;
}
//其他窗口创建之类的代码
if (m_hMutex) {
    CloseHandle(m_hMutex);
    m_hMutex = NULL;
}

3、窗口类中全局变量的作用域扩展:

extern UINT WM_ACTIVE_MSG;

4、窗口类中自定义消息的响应:

afx_msg LRESULT OnActiveMsg(WPARAM wParam, LPARAM lParam);
ON_REGISTERED_MESSAGE(WM_ACTIVE_MSG, &CDlgTestDlg::OnActiveMsg)
LRESULT CDlgTestDlg::OnActiveMsg(WPARAM wParam, LPARAM lParam)
{
    ::ShowWindow(m_hWnd, SW_NORMAL);
    ::SetForegroundWindow(m_hWnd);
    return TRUE;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值