WPF单实例启动

一、应用程序互斥锁

说明:测试反馈打包生成的程序,多次双击后运行多次,于是增加了单实例限制。

举个例子,我拿了唯一的钥匙打开了门,然后把钥匙带进去了,别人不能进,等我出去了关好门还要把钥匙还回去。

同理,通过静态构造函数static,在程序启动的那一刻,构造一个Mutex(互斥对象,系统级别的锁,跨进程可见),加上 typeof(App).GUID.ToString(),绑定唯一性标识 ,这样再次启动同一个程序的时候因为当前Mutex已经存在就直接退出。

关闭程序时需要释放当前Mutex

csharp 复制代码
public partial class App : Application
{
    private static readonly Mutex _mutex;
    private static bool _isSingleInstance;

    static App()
    {
        // 单实例检查
        string mutexName = "SIMWPF_SingleInstance_Mutex_" + typeof(App).GUID.ToString();
        _mutex = new Mutex(true, mutexName, out _isSingleInstance);

        if (!_isSingleInstance)
        {
            Environment.Exit(0);//立即终止进程,参数0标识正常退出,绕过清理不执行后续代码
            return;
        }
    }

    private void App_Startup(object sender, StartupEventArgs e)  //Application_Startup
    {
        // ... 原有代码 ...
        this.Exit += App_Exit;
    }
    
    private void App_Exit(object sender, ExitEventArgs e)
    {
        // 释放
        _mutex?.ReleaseMutex(); 
        _mutex?.Dispose();  //释放非托管资源(系统资源)
    }
}

附:

二、WPF应用程序的生命周期

三、 程序退出方法对比

方法 说明 适用场景
Environment.Exit(0) 立即终止进程 静态构造函数等早期退出
Application.Current.Shutdown() WPF 优雅退出 正常关闭程序
this.Close() 关闭窗口 关闭单个窗口

四、 其他

  1. lock是进程内的,无法跨进程检测,Mutex是系统级别的
  2. 静态构造函数是最早阶段执行的,确保单例检查在所有初始化之前完成
  3. 内核对象是由操作系统内核管理的资源,句柄是进程访问内核对象的"指针",引用计数是记录有多少句柄指向该对象
  4. 程序崩溃,操作系统会自动回收崩溃的进程内核对象(Mutex(互斥量)、Event(时间))、Semaphore(信号量)、File Handle(文件句柄)、Registry Key(注册表句柄)、Thread(线程)、Process(进程句柄)等)
相关推荐
光电大美美-见合八方中国芯1 小时前
用于无色波分复用光网络的 10.7 Gb/s 反射式电吸收调制器与半导体光放大器单片集成
网络·后端·ai·云计算·wpf·信息与通信·模块测试
晓纪同学1 小时前
WPF-02体系结构
wpf
晓纪同学2 小时前
WPF-01概述
wpf
海盗12345 小时前
OxyPlot 在 WPF 中的使用
.net·wpf
晓纪同学7 小时前
WPF-04 XAML概述
wpf
△曉風殘月〆1 天前
如何在WPF中捕获窗口外的事件
wpf
爱吃烤鸡翅的酸菜鱼2 天前
Java 事件发布-订阅机制全解析:从原生实现到主流中间件
java·中间件·wpf·事件·发布订阅
武藤一雄3 天前
WPF中ViewModel之间的5种通讯方式
开发语言·前端·microsoft·c#·wpf
CSharp精选营3 天前
都是微软亲儿子,WPF凭啥干不掉WinForm?这3个场景说明白了
c#·wpf·跨平台·winform
baivfhpwxf20233 天前
wpf TextBlock 控件如何根据内容换行?
wpf