最近有同事問道在應用程式啟動之後,再次雙擊應用程式,如何保證不再啟動新的應用程式,而是彈出之前已經啟動的進程,本質上這就是創建一個單實例的WPF應用程式。在VS的工程樹中有一個App.xaml和App.xaml.cs(這兩個文件都是VS自動生成的),在App.xaml.cs中定義了App類,該類繼承 ...
最近有同事問道在應用程式啟動之後,再次雙擊應用程式,如何保證不再啟動新的應用程式,而是彈出之前已經啟動的進程,本質上這就是創建一個單實例的WPF應用程式。在VS的工程樹中有一個App.xaml和App.xaml.cs(這兩個文件都是VS自動生成的),在App.xaml.cs中定義了App類,該類繼承自System.Windows.Application,從類的命名上也很容易看出這些類是跟應用程式的管理相關的,在講創建單實例應用程式之前,我們先瞭解一下Application這個類。
在上一篇WPF點滴(1) Main 函數中有部分對Application類的介紹,Application類封裝了WPF應用程式,並管理其生命周期,以下代碼用來創建一個Application:
Application app = new Application();
TestWindow window = new TestWindow();
app.MainWindow = window;
window.Show();
app.Run();
上面的代碼創建了一個Application,並將該Application的主視窗設置為TestWindow,可以將以上代碼簡化為同樣作用的以下代碼:
Application app = new Application();
TestWindow window = new TestWindow();
app.Run(window);
需要註意的是VS自動生成的App類中使用了另一種方法來初始化Application和主視窗,通過資源路徑來設置主視窗,
Application app = new Application();
app.StartupUri = new Uri("TestWindow.xaml", UriKind.Relative);
app.Run();
以上介紹瞭如何通過代碼啟動一個WPF應用程式,下麵介紹以下Application類的幾個常用場景:
1. 應用程式啟動後的初始化和退出前的清理,這裡會用到Startup和Exit這兩個事件,或者重載OnStartup和OnExit,這兩種方法是等效的,分別在程式啟動和退出時觸發;
2. 捕獲應用程式未處理的異常,DispatcherUnhandledException,異常處理的原則是就近捕獲,就近處理,這個事件更多是為了處理意料外的異常,防止程式奔潰的最後一層保護,因此不推薦過分依賴於這個事件中的異常處理。另外需要註意的是這個事件只會捕獲主線程(UI線程)中的異常,後臺線程中的未處理異常不會觸發該事件;
3. 在後臺線程中訪問UI,Application.Current.Dispatcher.Invoke(),這種方法比較簡單,另外一種方式是通過SynchronizationContext,這種方式更靈活一些。
現在終於可以切入正題了,如何創建創建單實例應用程式,前面說到了Startup這個事件,比較直接的方法是Startup的事件處理函數里判斷是否已經存在了另外一個已經啟動的應用程式,如果是就結束當前程式。這種方法可以實現單實例應用程式的效果,但是太暴力了,而且嚴格意義上並不是完全的單實例,應用程式畢竟已經啟動了,雖然又被幹掉了。
下麵介紹的是WPF的推薦方式,使用Windows窗體提供的內置支持,藉助WindowsFormsApplicationBase類,指定應用程式為單實例模式“IsSingleInstance = true”,
public class SingleInstanceApplication : WindowsFormsApplicationBase
{
private App m_App;
public SingleInstanceApplication()
{
IsSingleInstance = true;
}
protected override bool OnStartup(StartupEventArgs eventArgs)
{
m_App = new App
{
StartupUri = new System.Uri("MainWindow.xaml", System.UriKind.Relative)
};
m_App.Run();
return false;
}
protected override void OnStartupNextInstance(StartupNextInstanceEventArgs eventArgs)
{
base.OnStartupNextInstance(eventArgs);
}
}
重寫Main函數(具體方法參見WPF點滴(1) Main 函數),可以看到SingleInstanceApplication充當了App的wrapper,通過SingleInstanceApplication來啟動應用程式,並做單實例的的控制,SingleInstanceApplication.OnStartup()在應用程式首次啟動時觸發,SingleInstanceApplication.OnStartupNextInstance()在應用程式已經啟動並嘗試再次啟動時觸發。做這樣的設計是為了實現類似Word這樣的效果,Word只存在一個運行中的應用程式,每次雙擊並沒有啟動新的Word程式,只是打開了一個新視窗而已,而文檔的路徑就是通過“StartupEventArgs eventArgs”和“StartupNextInstanceEventArgs eventArgs”這兩個命令行參數來傳遞的。
public class StartUp
{
[STAThread]
public static void Main(string[] args)
{
SingleInstanceApplication application = new SingleInstanceApplication();
application.Run(args);
}
}
WPF沒有原生支持單實例模式,以上方法實際上是來自VB的一個設計特性,WindowsFormsApplicationBase的命名空間是Microsoft.VisualBasic.ApplicationServices,使用這個類需要添加對Microsoft.VisualBasic.dll的引用。