.Net平台应用程序单一运行实例代码实现

.Net 平台应用程序单一运行实例代码实现
作者:郑佐
日期:2006-7-2
 
概述
本文是针对 《基于.Net平台应用程序唯一运行实例实现》的补充,文章给出功能实现代码,其中SingleInstance类实现只允许一个实例运行,Program为测试主程序入口。在代码中标识说明文字。完整 代码下载。
 
主要代码
SingleInstance.cs 文件。
 
using System;
using System.IO;
using System.Diagnostics;
using System.Threading;
using System.Reflection;
using System.Runtime.InteropServices;
/*------------------------------------------------
  Zhengzuo 2006-07-01  http://blog.csdn.net/zhzuo       
--------------------------------------------------*/
namespace Zhengzuo.CSharpCode
{
    ///<summary>
    /// 只启动一个应用程序实例控制类
    ///</summary>
    public static class SingleInstance
    {
        private const int WS_SHOWNORMAL = 1;
        [DllImport("User32.dll")]
        private static extern bool ShowWindowAsync(IntPtr hWnd, int cmdShow);
        [DllImport("User32.dll")]
        private static extern bool SetForegroundWindow(IntPtr hWnd);
        // 标志文件名称
        private static string runFlagFullname = null;
        // 声明同步基元
        private static Mutex mutex = null;
       
        ///<summary>
        /// static Constructor
        ///</summary>
        static SingleInstance()
        {
        }
 
        #region api 实现
       
        ///<summary>
        /// 获取应用程序进程实例,如果没有匹配进程,返回Null
        ///</summary>
        ///<returns> 返回当前Process实例 </returns>
        public static Process GetRunningInstance()
        {           
            Process currentProcess = Process.GetCurrentProcess();// 获取当前进程
            // 获取当前运行程序完全限定名
            string currentFileName = currentProcess.MainModule.FileName;
            // 获取进程名为ProcessName的Process数组。
            Process[] processes = Process.GetProcessesByName(currentProcess.ProcessName);
            // 遍历有相同进程名称正在运行的进程
            foreach (Process process in processes)
            {
                if (process.MainModule.FileName == currentFileName)
                {
                    if (process.Id != currentProcess.Id)// 根据进程ID排除当前进程
                        return process;// 返回已运行的进程实例
                }
            }
            return null;
        }
 
        ///<summary>
        /// 获取应用程序句柄,设置应用程序前台运行,并返回bool值
        ///</summary>
        public static bool HandleRunningInstance(Process instance)
        {
            // 确保窗口没有被最小化或最大化
            ShowWindowAsync(instance.MainWindowHandle, WS_SHOWNORMAL);
            // 设置真实例程为foreground window
            return SetForegroundWindow(instance.MainWindowHandle);
        }
 
        ///<summary>
        /// 获取窗口句柄,设置应用程序前台运行,并返回bool值,重载方法
        ///</summary>
        ///<returns></returns>
        public static bool HandleRunningInstance()
        {
            Process p = GetRunningInstance();
            if (p != null)
            {
                HandleRunningInstance(p);
                return true;
            }
            return false;
        }
        #endregion
 
        #region Mutex 实现
        ///<summary>
        /// 创建应用程序进程Mutex
        ///</summary>
        ///<returns> 返回创建结果,true表示创建成功,false创建失败。 </returns>
        public static bool CreateMutex()
        {
            return CreateMutex(Assembly.GetEntryAssembly().FullName);
        }
 
        ///<summary>
        /// 创建应用程序进程Mutex
        ///</summary>
        ///<param name="name">Mutex 名称 </param>
        ///<returns> 返回创建结果,true表示创建成功,false创建失败。 </returns>
        public static bool CreateMutex(string name)
        {
            bool result = false;
            mutex = new Mutex(true, name, out result);
            return result;
        }
 
        ///<summary>
        /// 释放Mutex
        ///</summary>
        public static void ReleaseMutex()
        {
            if (mutex != null)
            {
                mutex.Close();
            }
        }
 
        #endregion
 
        #region 设置标志实现
        ///<summary>
        /// 初始化程序运行标志,设置成功,返回true,已经设置返回false,设置失败将抛出异常
        ///</summary>
        ///<returns> 返回设置结果 </returns>
        public static bool InitRunFlag()
        {
            if (File.Exists(RunFlag))
            {
                return false;
            }
            using (FileStream fs = new FileStream(RunFlag, FileMode.Create))
            {
            }
            return true;
        }
 
        ///<summary>
        /// 释放初始化程序运行标志,如果释放失败将抛出异常
        ///</summary>
        public static void DisposeRunFlag()
        {
            if (File.Exists(RunFlag))
            {
                File.Delete(RunFlag);
            }
        }
 
        ///<summary>
        /// 获取或设置程序运行标志,必须符合Windows文件命名规范
        /// 实现生成临时文件为依据,如果修改成设置注册表,那就不需要符合文件命名规范。
        ///</summary>
        public static string RunFlag
        {
            get
            {
                if(runFlagFullname == null)
                {
                    string assemblyFullName = Assembly.GetEntryAssembly().FullName;
                    //CommonApplicationData :"C://Documents and Settings//All Users//Application Data"
                    string path = Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData);
                    //"C://Program Files//Common Files"
                    //string path = Environment.GetFolderPath(Environment.SpecialFolder.CommonProgramFiles);
                    runFlagFullname = Path.Combine(path, assemblyFullName);
                }       
                return runFlagFullname;
            }
            set
            {
                runFlagFullname = value;
            }
        }
        #endregion
 
    }
}
 
Program.cs 文件。
 
using System;
using System.Windows.Forms;
using System.Diagnostics;
using Zhengzuo.CSharpCode;
 
namespace Zhengzuo.Test.WinGui
{
    static class Program
    {
        [STAThread]
        static void Main(string[] args)
        {
            if (args.Length == 0) // 没有传送参数
            {
                Process p = SingleInstance.GetRunningInstance();
                if (p != null) // 已经有应用程序副本执行
                {
                    SingleInstance.HandleRunningInstance(p);
                }
                else // 启动第一个应用程序
                {
                    RunApplication();
                }
            }
            else // 有多个参数
            {
                switch (args[0].ToLower())
                {
                    case "-api":
                        if (SingleInstance.HandleRunningInstance() == false)
                        {
                            RunApplication();
                        }                       
                        break;
                    case "-mutex":
                        if (args.Length >= 2) // 参数中传入互斥体名称
                        {
                            if ( SingleInstance.CreateMutex(args[1]) )
                            {
                                RunApplication();
                                SingleInstance.ReleaseMutex();
                            }
                            else
                            {
                                // 调用HandleRunningInstance()方法显示到前台。
                                MessageBox.Show(" 程序已经运行!" );
                            }
                        }
                        else
                        {
                            if (SingleInstance.CreateMutex())
                            {
                                RunApplication();
                                SingleInstance.ReleaseMutex();
                            }
                            else
                            {
                                // 调用HandleRunningInstance()方法显示到前台。
                                MessageBox.Show(" 程序已经运行!" );
                            }
                        }
                        break;
                    case "-flag":// 使用该方式需要在程序退出时调用
                        if (args.Length >= 2) // 参数中传入运行标志文件名称
                        {
                            SingleInstance.RunFlag = args[1];                           
                        }
                        try
                        {
                            if (SingleInstance.InitRunFlag())
                            {
                                RunApplication();
                                SingleInstance.DisposeRunFlag();
                            }
                            else
                            {
                                // 调用HandleRunningInstance()方法显示到前台。
                                MessageBox.Show(" 程序已经运行!" );
                            }
                        }
                        catch (Exception ex)
                        {
                            MessageBox.Show(ex.ToString());
                        }
                        break;
                    default:
                        MessageBox.Show(" 应用程序参数设置失败。" );
                        break;
                }
            }
        }
 
        // 启动应用程序
        static void RunApplication()
        {
            Application.EnableVisualStyles();
            Application.Run(new MainForm());
        }
    }
}
 
功能测试
功能测试类别包括下面五类,
1 .本地系统同一应用程序目录;
2 .本地系统同一应用程序修改运行文件名称使两次运行名称不同;
3 .本地系统两次运行程序目录不同,不修改文件名称;
4 .本地系统不同会话用户登录启动应用程序;
5 .远程计算机程序访问启动应用程序(一个程序在远程另一个在本地)。
 
运行CMD命令行,
第一种调用方式:
WindowsApplication1.exe
或 WindowsApplication1.exe –api
第二种 调用方式:
WindowsApplication1.exe –mutex
或WindowsApplication1.exe –mutex {F140AE26-626C-42f8-BD49-45025742205E}
第三种 调用方式:
WindowsApplication1.exe –flag
或WindowsApplication1.exe –flag c:/blog.csdn.net.zhzuo
 
测试结果
匹配/互斥/标志
1 同一目录
2 修改名称
3 不同目录
4 不同用户
5 远程访问
1 同一目录
O/O/O
 
 
 
 
2 修改名称
 
X/O/O
 
 
 
3 不同目录
 
 
X/O/O
 
 
4 不同用户
 
 
 
#/X/O
 
5 远程访问
 
 
 
 
X/O/O
备注:O - 表示成功,X – 表示失败,# - 程序第二个运行没有反应
 
针对远程访问的测试,需要在系统管理工具的.NET Framework 2.0 Configuration中进行设置授权该局域网路径允许访问,否则会抛出System.Security.SecurityException异常。
根据测试结果可见三种实现方式适用范围不同,理想的实现是结合他们的优点进行多点判断。
 
更多资源
关于.NET平台应用的开发,更多的技术文章可以访问http://blog.csdn.net/zhzuo,对于本文的建议或意见可在网站上留言。

你可能感兴趣的:(.Net平台应用程序单一运行实例代码实现)