C# · 12月 19, 2021

Winform防止程序重复运行

    需求:1、点击“关闭”按钮时,程序最小化到托盘,并没有退出,这时再次运行程序,不会重复运行,而是显示已运行的程序;2、支持不同目录;3、支持修改名称。

代码(不支持修改名称,不支持不同目录):

using System; System.Collections.Generic; System.IO; System.Linq; System.Text; System.Windows.Forms; Tool; System.Diagnostics; System.Reflection; System.Runtime.InteropServices;namespace 计算器{ static class Program { [DllImport(“user32.dll”)] public extern IntPtr FindWindow(string lpClassName,string lpWindowName); /// <summary> /// 该函数设置由不同线程产生的窗口的显示状态。 </summary> <param name=”hWnd”>窗口句柄</param> <param name=”cmdShow”>指定窗口如何显示。查看允许值列表,请查阅ShowWlndow函数的说明部分。<returns>如果函数原来可见,返回值为非零;如果函数原来被隐藏,返回值为零。</returns> [DllImport(User32.dllprivate extern bool ShowWindow(IntPtr hWnd,1)”>int cmdShow); 该函数将创建指定窗口的线程设置到前台,并且激活该窗口。键盘输入转向该窗口,并为用户改各种可视的记号。系统给创建前台窗口的线程分配的权限稍高于其他线程。 将被激活并被调入前台的窗口句柄。如果窗口设入了前台,返回值为非零;如果窗口未被设入前台,返回值为零。bool SetForegroundWindow(IntPtr hWnd); const int SW_SHOWNORMAL = 1; 应用程序的主入口点。 </summary> [STAThread] void Main() { Application.EnableVisualStyles(); Application.SetCompatibleTextRenderingDefault(false); Process processes = RunningInstance(); if (processes == null) { Application.Run(new Form1()); } else { HandleRunningInstance(processes); } } 获取正在运行的实例,没有运行的实例返回null; </summary> 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 ; } 显示已运行的程序。 HandleRunningInstance(Process instance) { try { IntPtr formHwnd = FindWindow(null,1)”>计算器); ShowWindow(formHwnd,SW_SHOWNORMAL); //显示 SetForegroundWindow(formHwnd); 放到前端 } catch (Exception ex) { Message@R_45_2419@.Show(ex.Message); } } }}View Code

代码(支持修改名称,支持不同目录):

Main() { Common.AutoRegister(); Application.EnableVisualStyles(); Application.SetCompatibleTextRenderingDefault(); createNew; using (System.Threading.Mutex m = new System.Threading.Mutex(true,Application.ProductName,1)”>out createNew)) { if (createNew) { FileOperator.SetValue(ProcessId进程ID写入文件 Application.Run( Form1()); } { { string strProcessId = FileOperator.GetValue(“); 从文件中获取进程ID int processId = Convert.ToInt32(strProcessId); Process process = Process.GetProcessById(processId); HandleRunningInstance(process); } { FileOperator.SetValue(进程ID写入文件 Application.Run( Form1()); } } } } (Exception ex) { Message@R_45_2419@.Show(ex.Message); } } }}View Code

    其实,IntPtr formHwnd = FindWindow(null,”计算器”); 这段代码是有BUG的,比如你打开一个名为“计算器”的文件夹,那么FindWindow找到的其实是这个文件夹,而不是计算器程序。我们可以在主窗体第一次显示的时候,记下窗口句柄,代码如下:

void Form1_Shown(object sender,EventArgs e){ FileOperator.SetValue(hwnd,Process.GetCurrentProcess().MainWindowHandle.ToString());}

    然后,显示已运行的程序时,从文件中读取之前记录的窗口句柄,代码如下:

<summary> 显示已运行的程序 HandleRunningInstance(Process instance){ { IntPtr hwnd = new IntPtr(Convert.ToInt32(FileOperator.GetValue())); ShowWindow(hwnd,SW_SHOWNORMAL); 显示 SetForegroundWindow(hwnd); } (Exception ex) { Message@R_45_2419@.Show(ex.Message); }}

     综上,再整理一下,就能得到完美的解决方案。