防止程式運行多個執行個體的方法有多種,如:通過使用互斥量和進程名等.而我想要實現的是:在程式運行多個執行個體時啟用的是第一個執行個體,使其獲得焦點,並在最上層顯示.
主要用到兩個API 函數:
- ShowWindowAsync 該函數設定由不同線程產生的視窗的顯示狀態。
- SetForegroundWindow 該函數將建立指定視窗的線程設定到前台,並且啟用該視窗。鍵盤輸入轉向該視窗,並為使用者改各種可視的記號。系統給建立前台視窗的線程分配的許可權稍高於其他線程。
代碼如下:
引用以下命名空間:
using System.Runtime.InteropServices;
using System.Diagnostics;
using System.Reflection;
//*****************************************************
static class Program
{
/// <summary>
/// 該函數設定由不同線程產生的視窗的顯示狀態。
/// </summary>
/// <param name="hWnd">視窗控制代碼</param>
/// <param name="cmdShow">指定視窗如何顯示。查看允許值列表,請查閱ShowWlndow函數的說明部分。</param>
/// <returns>如果函數原來可見,傳回值為非零;如果函數原來被隱藏,傳回值為零。</returns>
[DllImport("User32.dll")]
private static extern bool ShowWindowAsync(IntPtr hWnd, int cmdShow);
/// <summary>
/// 該函數將建立指定視窗的線程設定到前台,並且啟用該視窗。鍵盤輸入轉向該視窗,並為使用者改各種可視的記號。系統給建立前台視窗的線程分配的許可權稍高於其他線程。
/// </summary>
/// <param name="hWnd">將被啟用並被調入前台的視窗控制代碼。</param>
/// <returns>如果視窗設入了前台,傳回值為非零;如果視窗未被設入前台,傳回值為零。</returns>
[DllImport("User32.dll")]
private static extern bool SetForegroundWindow(IntPtr hWnd);
private const int WS_SHOWNORMAL = 1;
/// <summary>
/// 應用程式的主進入點。
/// </summary>
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Process instance = RunningInstance();
if (instance == null)
{
Form1 frm = new Form1();
Application.Run(new Form1());
}
else
{
HandleRunningInstance(instance);
}
}
/// <summary>
/// 擷取正在啟動並執行執行個體,沒有啟動並執行執行個體返回null;
/// </summary>
public static Process RunningInstance()
{
Process current = Process.GetCurrentProcess();
Process[] processes = Process.GetProcessesByName(current.ProcessName);
foreach (Process process in processes)
{
if (process.Id != current.Id)
{
if (Assembly.GetExecutingAssembly().Location.Replace("/", "\\") == current.MainModule.FileName)
{
return process;
}
}
}
return null;
}
/// <summary>
/// 顯示已啟動並執行程式。
/// </summary>
public static void HandleRunningInstance(Process instance)
{
ShowWindowAsync(instance.MainWindowHandle, WS_SHOWNORMAL); //顯示,可以注釋掉
SetForegroundWindow(instance.MainWindowHandle); //放到前端
}
}