之前也做過這樣的單例處理 但是不能激活程序並顯示在最前端 今天看了Charles Chen 的C#程序只允許運行一個實例的解決方案 中文章的方案 實現了單例運行並激活序窗口 方法和Charles Chen的一樣 只是做了些細節的優化。
public class SoftHelper { ///<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 SW_SHOWNOMAL = 1; private static void HandleRunningInstance(Process instance) { ShowWindowAsync(instance.MainWindowHandle, SW_SHOWNOMAL);//顯示 SetForegroundWindow(instance.MainWindowHandle);//當到最前端 } private static Process RuningInstance() { Process currentProcess = Process.GetCurrentProcess(); Process[] Processes = Process.GetProcessesByName(currentProcess.ProcessName); foreach (Process process in Processes) { if (process.Id != currentProcess.Id) { return process; } } return null; } /// <summary> /// 程序以單例運行 /// </summary> public static void SoftSingle<T>() where T:Form, new() { Process process = RuningInstance(); if(process==null) { var mainForm = new T(); Application.Run(mainForm); } else { HandleRunningInstance(process); } } }
調用為:SoftHelper.SoftSingle<Main>();