C++ MFC 只启动一个程序实例 唤醒之前的实例(完整源码)

初级代码游戏的专栏介绍与文章目录-CSDN博客

很多时候我们希望只允许启动一个程序实例,如果再次运行,就唤醒之前的实例。

目录

[1 概述](#1 概述)

[2 相关技术介绍](#2 相关技术介绍)

[2.1 互斥对象](#2.1 互斥对象)

[2.2 查找窗口](#2.2 查找窗口)

[2.3 唤醒窗口](#2.3 唤醒窗口)


1 概述

技术上并不难,涉及到以下几个技术细节:

  1. 使用互斥对象,确保检查程序是否已经启动时没有并发错误。
  2. 查找之前已经运行的程序,可以通过窗口名称查找,也可以通过共享文件交互
  3. 控制之前已经运行的程序,可以通过发送窗口消息,也可以通过共享文件交互

窗口程序通过窗口机制,查找和发消息都是操作系统的标准功能,如果 使用了配置文件,通过文件来交互也是可以的,只不过要定时查询,实时性没有发消息好。

2 相关技术介绍

下面介绍相关代码:

2.1 互斥对象

cpp 复制代码
	HANDLE hOneInstance;
	hOneInstance = ::CreateMutex(NULL, FALSE, L"自定义的名字");
	if (GetLastError() == ERROR_ALREADY_EXISTS)
	{
        //在这里处理已经存在的情形
        //查找窗口、唤醒窗口,然后退出
    }
    //正常启动

互斥对象是系统对象,本例中我们并不需要更多功能,只要检测创建是否成功即可。因为,互斥对象所有句柄被释放后就会被销毁,程序不退出,下一个程序创建会失败,程序退出后下一个就可以成功创建。对于我们的目的而言锁定互斥对象是不必要的。

CreateMutes原型如下:

cpp 复制代码
HANDLE CreateMutexA(
  [in, optional] LPSECURITY_ATTRIBUTES lpMutexAttributes,
  [in]           BOOL                  bInitialOwner,
  [in, optional] LPCSTR                lpName
);

第一个参数涉及到权限,可以设置为NULL,第二个参数指示创建完成是不是要立即锁定,第三个参数是名字,可以不设置,设置的话不能和已经存在的内核对象同名,所有种类的内核对象共享同一个名字空间。

2.2 查找窗口

cpp 复制代码
		HWND hWnd = FindWindow(NULL, TEXT("窗口标题"));
		if (IsWindow(hWnd))
		{
			//遍历处理所有同名窗口!!!
			while(IsWindow(hWnd))
			{
				//在这里处理窗口

                //查找下一个
				hWnd = FindWindowEx(NULL,hWnd,NULL,TEXT("窗口标题"));
			}
		}

查找窗口的关键点是窗口标题必须正确,所以最好把窗口标题单独定义出来,并在窗口初始化里面专门设定一次,而且不要在程序里面修改窗口标题。这其实没法从技术上预防,程序的任何地方都可以随意修改窗口标题,如果没法掌控全部源代码,那就只好走文件控制的方式了。

文件控制就是把窗口句柄写入文件,每次启动去文件里面读,读不到就是还没有程序实例在运行。

这里写成循环是因为理论上可能有相同标题的窗口,需要这两个函数配合才能遍历所有同名窗口。

cpp 复制代码
HWND FindWindowA(
  [in, optional] LPCSTR lpClassName,
  [in, optional] LPCSTR lpWindowName
);

FindWindow用窗口类名和/或窗口标题查找窗口,大部情况下我们只用窗口标题来查找。

cpp 复制代码
HWND FindWindowExA(
  [in, optional] HWND   hWndParent,
  [in, optional] HWND   hWndChildAfter,
  [in, optional] LPCSTR lpszClass,
  [in, optional] LPCSTR lpszWindow
);

FindWindowEx查找子窗口。

2.3 唤醒窗口

cpp 复制代码
				// 显示
				if(!::ShowWindow(hWnd, SW_NORMAL))
				{
                    //出错
				} 
				// 激活
				if(!::SetForegroundWindow(hWnd))
				{
                    //出错
				}

这两句代码嵌入到之前查找窗口的循环里面即可将之前的实例的窗口显示到前端。

cpp 复制代码
BOOL SetForegroundWindow(
  [in] HWND hWnd
);

这个函数的效果是把窗口放到前端并激活,输入焦点将设置到这个窗口中。

如果想让窗口的标题栏闪烁,用这个API即可:

cpp 复制代码
BOOL FlashWindow(
  [in] HWND hWnd,
  [in] BOOL bInvert
);

第二个参数为FALSE将保持之前的活动或非活动状态,为TRUE则切换状态。这个调用只闪烁一次,想多闪烁几次要用定时器来控制(不要用Sleep,会导致界面僵死)。

(这里是文档结束)

相关推荐
羊小猪~~44 分钟前
数据结构C语言描述2(图文结合)--有头单链表,无头单链表(两种方法),链表反转、有序链表构建、排序等操作,考研可看
c语言·数据结构·c++·考研·算法·链表·visual studio
脉牛杂德1 小时前
多项式加法——C语言
数据结构·c++·算法
legend_jz1 小时前
STL--哈希
c++·算法·哈希算法
CSUC2 小时前
【C++】父类参数有默认值时子类构造函数列表中可以省略该参数
c++
Vanranrr2 小时前
C++ QT
java·c++·qt
鸿儒5172 小时前
C++ lambda 匿名函数
开发语言·c++
van叶~3 小时前
算法妙妙屋-------1.递归的深邃回响:二叉树的奇妙剪枝
c++·算法
knighthood20013 小时前
解决:ros进行gazebo仿真,rviz没有显示传感器数据
c++·ubuntu·ros
半盏茶香3 小时前
【C语言】分支和循环详解(下)猜数字游戏
c语言·开发语言·c++·算法·游戏
小堇不是码农4 小时前
在VScode中配置C_C++环境
c语言·c++·vscode