Force Single Instance with Mutex handling restart application

前端 未结 4 1978
[愿得一人]
[愿得一人] 2021-01-25 12:04

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

相关标签:
4条回答
  • 2021-01-25 12:16

    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());
        }
    
    0 讨论(0)
  • 2021-01-25 12:16

    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();
            }
    
    0 讨论(0)
  • 2021-01-25 12:24

    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;
        }
    }
    
    0 讨论(0)
  • 2021-01-25 12:25

    I'm not sure what exactly goes wrong in your case but some issues might be:

    • You should probably use try finally to close the mutex to maximize changes of proper cleanup
    • In case cleanup failed previous time then Mutex.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.
    • You are giving the Mutex 5 seconds to get aquired. If your shutdown will take over 5 seconds then the next startup will fail. A better option might be to release mutex right before initiating shutdown. (Provided of course that the 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();
            }
        }
    }
    
    0 讨论(0)
提交回复
热议问题