
先看下未限制前的exe程序多开效果:
然后再看下限制后的exe程序还能不能多开,结果当然是一直都一个,无法再多开了:
那么这种限制只能启动一个exe程序是如何实现的呢?
这里涉及到了进程间的同步问题,同一个应用程序启动后,不能再开启该程序的新进程。
它用到了一个类:Mutex,定义如下:
这个互斥类的作用是用于进程间对系统共享资源的独占访问操作系统特定对象时而设计的。
Mutex instance = new Mutex(true, “MutexName”, out bool createdNew);
编写上述代码,创建互斥类Mutex的一个实例。
第1个参数initiallyOwned为true,标记调用线程是否具有初始所有权。
第2个参数name是起一个名字,如果为空,则不生效,所以要起名名字。系统会判别进程中是否有这个名称的对象,若有,则不允许再开新进程。
第3个参数createdNew指示,是否允许开启新进程。你可以根据它的返回值true或false做自己的业务逻辑。
如果为true,说明就可以开启新进程,如果为false说明已经存在一个进程了,就没必须再去创建了。
完整代码如下:
using System;
using System.Threading;
using System.Windows.Forms;
namespace Practice.Client
{
static class Program
{
///
/// 应用程序的主入口点。
///
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Mutex instance = new Mutex(true, "MutexName", out bool createdNew);
if (createdNew)
{
Application.Run(new MainForm());
instance.ReleaseMutex();
}
else
{
Application.Exit();
}
}
}
}
代码参考: C#代码 程序只能运行一次!
防止程序运行多个实例的方法有多种,如:通过使用互斥量和进程名等.而我想要实现的是:在程序运行多个实例时激活的是第一个实例,使其获得焦点,并在前端显示.
主要用到两个API 函数:
ShowWindowAsync 该函数设置由不同线程产生的窗口的显示状态。
SetForegroundWindow 该函数将创建指定窗口的线程设置到前台,并且激活该窗口。键盘输入转向该窗口,并为用户改各种可视的记号。系统给创建前台窗口的线程分配的权限稍高于其他线程。
代码如下:
static class Program { /// /// 该函数设置由不同线程产生的窗口的显示状态。 /// /// 窗口句柄 /// 指定窗口如何显示。查看允许值列表,请查阅ShowWlndow函数的说明部分。 /// 如果函数原来可见,返回值为非零;如果函数原来被隐藏,返回值为零。 [DllImport(“User32.dll”)] private static extern bool ShowWindowAsync(IntPtr hWnd, int cmdShow); /// /// 该函数将创建指定窗口的线程设置到前台,并且激活该窗口。键盘输入转向该窗口,并为用户改各种可视的记号。系统给创建前台窗口的线程分配的权限稍高于其他线程。 /// /// 将被激活并被调入前台的窗口句柄。 /// 如果窗口设入了前台,返回值为非零;如果窗口未被设入前台,返回值为零。 [DllImport(“User32.dll”)] private static extern bool SetForegroundWindow(IntPtr hWnd); private const int WS_SHOWNORMAL = 1; /// /// 应用程序的主入口点。 /// [STAThread] static void Main() { Application.EnableVisualStyles(); Application.SetCompatibleTextRenderingDefault(false); Process instance = RunningInstance(); if (instance == null) { Form1 frm = new Form1(); Application.Run(new Form1()); } else { HandleRunningInstance(instance); } } /// /// 获取正在运行的实例,没有运行的实例返回null; /// public static Process RunningInstance() { Process current = Process.GetCurrentProcess(); Process[] processes = Process.GetProcessesByName(current.ProcessName); foreach (Process process in processes) { if (process.Id != current.Id) { if (Assembly.GetExecutingAssembly().Location.Replace(“/”, “//”) == current.MainModule.FileName) { return process; } } } return null; } /// /// 显示已运行的程序。 /// public static void HandleRunningInstance(Process instance) { ShowWindowAsync(instance.MainWindowHandle, WS_SHOWNORMAL); //显示,可以注释掉 SetForegroundWindow(instance.MainWindowHandle); //放到前端 } }
记着引用以下命名空间:
using System.Runtime.InteropServices;
using System.Diagnostics;
using System.Reflection;
http://t.csdn.cn/aWkPj