如果應(yīng)用程序的另一個(gè)實(shí)例影響到可選(非首要)功能,應(yīng)用程序啟動(dòng)時(shí)必須:
1)檢測(cè)是否有用戶正在運(yùn)行該應(yīng)用程序。
2)阻止所有有問(wèn)題的功能。
3)通知當(dāng)前用戶無(wú)法使用特定功能的原因。
如果應(yīng)用程序的另一個(gè)實(shí)例影響首要功能,同樣,您的應(yīng)用程序必須:
1)檢測(cè)是否有用戶正在運(yùn)行該應(yīng)用程序。
2)向當(dāng)前用戶報(bào)告錯(cuò)誤情況,然后退出。
下面給出一個(gè)實(shí)例:
創(chuàng)建 Win32 應(yīng)用程序
啟動(dòng) Visual Studio 并新建一個(gè)名為 FastUserSwitching 的 Win32 應(yīng)用程序。
Visual C++ 6.0 用戶: 從可用項(xiàng)目類型列表中選擇 Win32 應(yīng)用程序,然后在應(yīng)用程序安裝向?qū)е羞x擇一個(gè)典型的“Hello World”應(yīng)用程序。
Visual Studio .NET 用戶: 在 Visual C++ 項(xiàng)目中選擇 Win32 項(xiàng)目并接受應(yīng)用程序安裝向?qū)е酗@示的默認(rèn)應(yīng)用程序設(shè)置。
添加接收會(huì)話切換通知的代碼
如果你的應(yīng)用程序需要知道何時(shí)要在活動(dòng)用戶會(huì)話中運(yùn)行以及何時(shí)發(fā)生了會(huì)話切換,該應(yīng)用程序可以通過(guò)調(diào)用 WTSRegisterSessionNotification 函數(shù)進(jìn)行注冊(cè)以接收 WM_WTSSESSION_CHANGE 消息:
1、打開(kāi) stdafx.h 并在包含 windows.h 的語(yǔ)句之前添加以下 #define 語(yǔ)句:
#define _WIN32_WINNT 0x0501
這是 winuser.h 的要求,其目的是定義通知類型和宏。
2、在 FastUserSwitching.cpp 的頂部包含以下頭文件(其中包含 WTSRegisterSessionNotification 函數(shù)原型):
#include <wtsapi32.h>
3、將 Wtsapi32.lib 添加到項(xiàng)目的庫(kù)列表。
4、在 FastUserSwitching.cpp 中找到 InitInstance 函數(shù)。在函數(shù)的尾部的 return 語(yǔ)句之前,添加對(duì) WTSRegisterSessionNotification 的調(diào)用,如下所示:
WTSRegisterSessionNotification(hWnd, NOTIFY_FOR_THIS_SESSION);
5、找到 WndProc 窗口過(guò)程并添加處理 WM_WTSSESSION_CHANGE 消息的 case 語(yǔ)句。 此消息的 wParam 包含狀態(tài)編碼,表明發(fā)出會(huì)話更改通知的原因。 添加以下代碼檢測(cè)可用狀態(tài)編碼的子集并顯示消息框,表明已收到哪些狀態(tài)編碼:

code
1
case WM_WTSSESSION_CHANGE:
2
switch( wParam )
3
{
4
case WTS_CONSOLE_CONNECT:
5
MessageBox(hWnd, TEXT("WTS_CONSOLE_CONNECT"),
6
TEXT("WM_WTSSESSION_CHANGE"), MB_OK );
7
break;
8
case WTS_CONSOLE_DISCONNECT:
9
MessageBox(hWnd, TEXT("WTS_CONSOLE_DISCONNECT"),
10
TEXT("WM_WTSSESSION_CHANGE"), MB_OK );
11
break;
12
case WTS_SESSION_LOCK:
13
MessageBox(hWnd, TEXT("WTS_SESSION_LOCK"),
14
TEXT("WM_WTSSESSION_CHANGE"), MB_OK );
15
break;
16
case WTS_SESSION_UNLOCK:
17
MessageBox(hWnd, TEXT("WTS_SESSION_UNLOCK"),
18
TEXT("WM_WTSSESSION_CHANGE"), MB_OK );
19
break;
20
default:
21
break;
22
}
23
break;
24
6、每一個(gè)對(duì) WTSRegisterSessionNotification 的調(diào)用應(yīng)與一個(gè)對(duì) WTSUnRegisterSessionNotification 的調(diào)用匹配。 在 WndProc 中修改 WM_DESTROY 消息的處理,如下所示:
case WM_DESTROY:
WTSUnRegisterSessionNotification(hWnd);
PostQuitMessage(0);
break;
確認(rèn)會(huì)話切換通知
本任務(wù)假定至少有兩個(gè)用戶帳戶。 如果您只有一個(gè)帳戶,請(qǐng)?jiān)傩陆ㄒ粋€(gè)帳戶。
1、重新生成項(xiàng)目。
2、運(yùn)行該應(yīng)用程序。
3、從開(kāi)始菜單中,單擊注銷,然后單擊切換用戶。
4、單擊當(dāng)前用戶名返回到前一個(gè)用戶會(huì)話。
5、確認(rèn)您已經(jīng)收到 WTS_SESSION_LOCK 和 WTS_SESSION_UNLOCK 通知。
6、單擊確定可消除這兩個(gè)消息框。
7、從開(kāi)始菜單中,單擊注銷,然后單擊切換用戶。
8、切換到新用戶會(huì)話,然后再切換回原來(lái)的用戶會(huì)話。
9、確認(rèn)您已經(jīng)收到 WTS_SESSION_LOCK、WTS_CONSOLE_DISCONNECT、WTS_SESSION_UNLOCK 和 WTS_CONSOLE_CONNECT 通知。
10、單擊確定可消除所有消息框。
11、關(guān)閉應(yīng)用程序。
檢測(cè)現(xiàn)有應(yīng)用程序?qū)嵗?/h3>
若要檢測(cè)現(xiàn)有的應(yīng)用程序?qū)嵗褂靡粋€(gè)全局 mutex 或 semaphore 對(duì)象(名稱已知)。在對(duì)象名前添加前綴“Global\”確保使用全局命名空間。這樣就可以檢測(cè)在不同用戶會(huì)話環(huán)境中運(yùn)行的您的應(yīng)用程序?qū)嵗?nbsp;
使用 FindWindow 或 FindWindowEx 的傳統(tǒng)方法在啟用快速用戶切換的 Windows XP 系統(tǒng)中不起作用,因?yàn)檫@些方法不會(huì)檢測(cè)在不同用戶會(huì)話環(huán)境中(或不同桌面)運(yùn)行的應(yīng)用程序?qū)嵗?nbsp;
1、編輯 FastUserSwitching.cpp。
2、在文件頂部的現(xiàn)有全局變量后聲明并初始化一個(gè)全局變量,存儲(chǔ) mutex 對(duì)象的句柄。
HANDLE g_hMutexAppRunning = NULL;
3、為新建函數(shù)添加以下函數(shù)原型,檢測(cè)應(yīng)用程序?qū)嵗欠褚汛嬖冢?nbsp;
BOOL AppInstanceExists();
4、在源文件的末尾,使用以下代碼創(chuàng)建 AppInstanceExists 函數(shù)。 此代碼試圖創(chuàng)建一個(gè)全局 mutex 對(duì)象,然后檢查是否創(chuàng)建并打開(kāi)了 mutex 對(duì)象(通過(guò)檢查錯(cuò)誤代碼 ERROR_ALREADY_EXISTS 實(shí)現(xiàn))。 在這種情況下,錯(cuò)誤代碼表明已有應(yīng)用程序?qū)嵗\(yùn)行。 如果是這樣,代碼關(guān)閉 mutex 對(duì)象并返回“TRUE”。 如果此函數(shù)成功創(chuàng)建了一個(gè)新的 mutex 對(duì)象,將返回“FALSE”,表明這是第一個(gè)應(yīng)用程序?qū)嵗?nbsp;

code
1
BOOL AppInstanceExists()
2

{
3
BOOL bAppRunning = FALSE;
4
// Create a global mutex. Use a unique name, for example
5
// incorporating your company and application name.
6
g_hMutexAppRunning = CreateMutex( NULL, FALSE, "Global\\My MpApp.EXE");
7
// Check if the mutex object already exists, indicating an
8
// existing application instance
9
if (( g_hMutexAppRunning != NULL ) &&
10
( GetLastError() == ERROR_ALREADY_EXISTS))
11
{
12
// Close the mutex for this application instance. This assumes
13
// the application will inform the user that it is
14
// about to terminate
15
CloseHandle( g_hMutexAppRunning );
16
g_hMutexAppRunning = NULL;
17
}
18
// Return False if a new mutex was created,
19
// as this means it's the first app instance
20
return ( g_hMutexAppRunning == NULL );
21
}
5、您必須確保當(dāng)運(yùn)行的應(yīng)用程序終止時(shí),mutex 對(duì)象關(guān)閉。 將以下代碼添加到 WinMain 函數(shù)的末尾,位于消息循環(huán)之后,最后的 return 語(yǔ)句之前:
if (g_hMutexAppRunning != NULL )
{
CloseHandle(g_hMutexAppRunning);
g_hMutexAppRunning = NULL;
}
將現(xiàn)有應(yīng)用程序?qū)嵗O(shè)置到前臺(tái)
如果只允許運(yùn)行應(yīng)用程序的一個(gè)實(shí)例,您應(yīng)當(dāng)使用 FindWindow 和 SetForegroundWindow API 在后續(xù)實(shí)例啟動(dòng)時(shí)將現(xiàn)有實(shí)例置于前臺(tái)(如果現(xiàn)有實(shí)例運(yùn)行在當(dāng)前用戶會(huì)話中)。 您必須測(cè)試 FindWindow 的返回值,因?yàn)槿绻F(xiàn)有應(yīng)用程序?qū)嵗诹硪粋€(gè)用戶的會(huì)話中運(yùn)行,將返回 NULL。
找到 InitInstance 函數(shù)進(jìn)行修改,如下所示:

code
1
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
2
{
3
HWND hWnd;
4
hInst = hInstance;
5
// Check if another application instance is already running
6
if ( AppInstanceExists() == TRUE )
7
{
8
HWND hWndOtherInstance;
9
hWndOtherInstance = FindWindow(szWindowClass, szTitle);
10
if ( hWndOtherInstance != (HWND)NULL )
11
{
12
// Application is running in current user's session
13
if ( IsIconic(hWndOtherInstance) )
14
ShowWindow(hWndOtherInstance, SW_RESTORE);
15
SetForegroundWindow(hWndOtherInstance);
16
}
17
else
18
{
19
MessageBox(NULL, TEXT("An instance of this app is running in another user's session"), szTitle, MB_OK);
20
}
21
return FALSE;
22
}
23
hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
24
CW_USEDEFAULT, , CW_USEDEFAULT, , NULL, NULL,
25
hInstance, NULL );
26
if (!hWnd)
27
return FALSE;
28
29
ShowWindow(hWnd, nCmdShow);
30
UpdateWindow(hWnd);
31
WTSRegisterSessionNotification(hWnd, NOTIFY_FOR_THIS_SESSION);
32
return TRUE;
33
}
測(cè)試應(yīng)用程序檢測(cè)
1、生成項(xiàng)目。
2、運(yùn)行該應(yīng)用程序。
3、最小化應(yīng)用程序。
4、啟動(dòng)應(yīng)用程序的另一個(gè)實(shí)例,檢查現(xiàn)有應(yīng)用程序是否被恢復(fù)并置于前臺(tái)。
5、反復(fù)啟動(dòng)其他的啟動(dòng)應(yīng)用程序?qū)嵗⒋_保每次啟動(dòng)時(shí)現(xiàn)有應(yīng)用程序都置于前臺(tái)。
6、在應(yīng)用程序的一個(gè)實(shí)例運(yùn)行時(shí),切換到新的用戶會(huì)話。
7、試圖啟動(dòng)應(yīng)用程序,您會(huì)看到一個(gè)消息框,它說(shuō)明了該應(yīng)用程序已在另一個(gè)用戶會(huì)話中運(yùn)行。
8、單擊確定消除此消息框。
9、返回到原來(lái)的用戶會(huì)話,關(guān)閉會(huì)話切換通知消息窗口并退出應(yīng)用程序