代码改变世界

C#只允许运行应用程序的一个实例的正确写法

2011-04-27 11:40  zhoujie  阅读(1359)  评论(1编辑  收藏  举报

有时我们需要只允许运行应用程序的一个实例,当进程启动时,如果发现应用程序的一个实例在运行,就自动停止运行。我们通常通过Mutex互斥体在Main函数中实现,通常的写法是:

[STAThread]
static void Main()
{
    
bool createNew;
    
using (System.Threading.Mutex m = new System.Threading.Mutex(true, Application.ProductName, out createNew))
    {
        
if (createNew)
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(
false);
            Application.Run(
new Form1());
        }
        
else
        {
            MessageBox.Show(
"Only one instance of this application is allowed!");
        }
    }
}

 我们在寻找答案时,往往来去匆匆,根本不去理会Mutex的特性和注意事项。经过简单的测试,OK,拿来就用。此时我们忽略了一个重要的前提条件:Mutex的命名规则。以上的写法在单用户下运行没有问题;在多用户下,每个用户都能启动一个实例,也就不能保证单实例运行了。

如果需要在终端机服务器上使用,并且只允许一个实例的话,请使用下面的写法:

[STAThread]
        
static void Main()
        {
            
bool createNew;
            
try
            {
                
using (System.Threading.Mutex m = new System.Threading.Mutex(true"Global\\" + Application.ProductName, out createNew))
                {
                    
if (createNew)
                    {
                        Application.EnableVisualStyles();
                        Application.SetCompatibleTextRenderingDefault(
false);
                        Application.Run(
new Form1());
                    }
                    
else
                    {
                        MessageBox.Show(
"Only one instance of this application is allowed!");
                    }
                }
            }
            
catch
            {
                MessageBox.Show(
"Only one instance of this application is allowed!");
            }
        }

 

 以下是MSDN的说明:

 

在运行终端服务的服务器上,已命名的系统 mutex 可以具有两级可见性。如果名称以前缀“Global\”开头,则 mutex 在所有终端服务器会话中均为可见。如果名称以前缀“Local\”开头,则 mutex 仅在创建它的终端服务器会话中可见。在这种情况下,服务器上各个其他终端服务器会话中都可以拥有一个名称相同的独立 mutex。如果创建已命名 mutex 时不指定前缀,则它将采用前缀“Local\”。在终端服务器会话中,只是名称前缀不同的两个 mutex 是独立的 mutex,这两个 mutex 对于终端服务器会话中的所有进程均为可见。即:前缀名称“Global\”和“Local\”说明 mutex 名称相对于终端服务器会话(而并非相对于进程)的范围。