I have a problem when i want use mutex to force single instance of my program.
I a winform App with a WebBrowser Control. I need to autorestart if certain conditions ar
Try using try..finally
for managing the Mutex.
I usually use this approach. Worked well in combination with Application.Restart()
.
private const string ApplicationMutexName = "<myprogramkey>";
/// <summary>
/// The main entry point for the application.
/// </summary>
[STAThread]
public static void Main()
{
RunApplicationPreservingSingleInstance();
}
private static void RunApplicationPreservingSingleInstance()
{
Mutex mutex = AcquireMutex();
if (mutex == null)
{
return;
}
try
{
RunApplication();
}
finally
{
mutex.ReleaseMutex();
}
}
private static Mutex AcquireMutex()
{
Mutex appGlobalMutex = new Mutex(false, ApplicationMutexName);
if (!appGlobalMutex.WaitOne(3000))
{
return null;
}
return appGlobalMutex;
}
private static void RunApplication()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new MainForm());
}
I do:
bool onlyInstance = false;
mutex = new System.Threading.Mutex(true, "qFluid", out onlyInstance);
int cntrSec = 0;
//per far funzionare restart ,aspetto per 3 sec per vedere se va via servizio
while (!onlyInstance & cntrSec < 3)
{
System.Threading.Thread.Sleep(1000);
cntrSec += 1;
}
if (!onlyInstance)
{
Xceed.Wpf.Toolkit.MessageBox.Show("The application is already arunning");
App.Current.Shutdown();
}
Single instance apps are well supported by the framework. It supports goodies such as disabling it on-the-fly, what you need here, and getting a notification when another instance is started. You'll want to use that to give your first instance the focus. Rewrite your Program.cs code like this:
using System;
using System.Windows.Forms;
using Microsoft.VisualBasic.ApplicationServices; // NOTE: add reference to Microsoft.VisualBasic
namespace WindowsFormsApplication1 {
class Program : WindowsFormsApplicationBase {
public Program() {
EnableVisualStyles = true;
MainForm = new Form1();
IsSingleInstance = true;
}
public static void Main(string[] args) {
Instance = new Program();
Instance.Run(args);
}
protected override void OnStartupNextInstance(StartupNextInstanceEventArgs eventArgs) {
// Nicety, focus the single instance.
Instance.MainForm.Activate();
}
public static void Restart() {
// What you asked for. Use Program.Restart() in your code
Instance.IsSingleInstance = false;
Application.Restart();
}
private static Program Instance;
}
}
I'm not sure what exactly goes wrong in your case but some issues might be:
try finally
to close the mutex to maximize changes of proper cleanupMutex.WaitOne
will throw an AbandonedMutexException
instead of return true
. In this case your code will exit with an error box while you can just continue.Mutex
is there only for user convenience).example
the following example will require that you use Program.Restart
to restart the application. Else the time window can still be the problem.
static class Program
{
private static Mutex _mutex;
[STAThread]
static void Main()
{
_mutex = new Mutex(false, "myprogramkey");
try
{
try
{
if (!_mutex.WaitOne(1))
{
_mutex.Dispose();
_mutex = null;
MessageBox.Show("Error!");
return;
}
}
catch (AbandonedMutexException) { /* Mutex wasn't property released last time.*/ }
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
finally
{
if (_mutex != null)
{
_mutex.ReleaseMutex();
_mutex.Dispose();
}
}
}
public static void Restart()
{
if (_mutex != null)
{
_mutex.ReleaseMutex();
_mutex.Dispose();
_mutex = null;
Application.Restart();
}
}
}