C#让应用程序只运行一个实例的几种方法
一 判断是否有相同的实例已经运行
1 根据“Mutex”判断是否有相同的实例在运行
public bool IsRunningProcessByMutex()
{
bool createNew;
using (System.Threading.Mutex mutex = new System.Threading.Mutex(true, Application.ProductName, out createNew))
{
return !createNew;
}
}
特点:不能返回前一个(已经运行的)实例的 Process
2 根据进“程名称”判断是否有相同的实例在运行,如果已运行,则返回已运行的实例
/// <returns>已有实例运行返回true,否则为false</returns>
private static bool GetRunningProcessByProcessName(out Process runningProcess)
{
bool returnValue = false;
runningProcess = null;
Process current = Process.GetCurrentProcess();
Process[] processes = Process.GetProcessesByName(current.ProcessName);
foreach (Process process in processes)
{
if (process.Id != current.Id)
{
if (process.ProcessName == current.ProcessName)
{
runningProcess = process;
returnValue = true;
break;
}
}
}
return returnValue;
}
特点:1)不同程序有可能有相同的进程名;
2)如果运行了一个实例后修改文件名再运行另外一个实例,此方法失效。
3 根据“进程名称和路径”判断是否有相同的实例在运行,如果已运行,则返回已运行的实例
{
//... 除进程变焦外,其它同方法2
if (process.MainModule.FileName ==
Assembly.GetExecutingAssembly().Location.Replace("/", "\\"))
}
特点:修改了文件名或改变了文件路径,此方法失效。
4 根据“程序集的签名”判断是否有相同的实例在运行,如果已运行,则返回已运行的实例
/// <returns>已有实例运行返回true,否则为false</returns>
private static bool GetRunningProcessByAssemblyName(out Process runningProcess)
{
bool returnValue = false;
runningProcess = null;
AssemblyName currentAssemblyName =
AssemblyName.GetAssemblyName(Assembly.GetExecutingAssembly().Location);
AssemblyName processAssemblyName = new AssemblyName();
Process current = Process.GetCurrentProcess();
Process[] processes = Process.GetProcesses();
foreach (Process process in processes)
{
// 排除一些其他进程,可以加快点速度。
if (process.Id != current.Id &&
process.ProcessName != "System" &&
process.ProcessName != "csrss" &&
process.ProcessName != "svchost" &&
process.ProcessName != "services" &&
process.ProcessName != "smss" &&
process.ProcessName != "winlogon" &&
process.ProcessName != "explorer" &&
process.ProcessName != "pds" &&
process.ProcessName != "alg" &&
process.ProcessName != "msdtc" &&
process.ProcessName != "spoolsv" &&
process.ProcessName != "lsass" &&
process.ProcessName != "Idle" &&
process.ProcessName != "iexplore" &&
process.ProcessName != "sqlserver" &&
process.ProcessName != "notepad" &&
process.ProcessName != "360tray" &&
process.ProcessName != "XDict"
)
{
try
{
// 获取文件的程序集
processAssemblyName = AssemblyName.GetAssemblyName(process.MainModule.FileName);
}
catch (Exception)
{
processAssemblyName = null;
}
// 通过 GetPublicKey() 来获取程序集的公钥;需要对程序集签名,否则 GetPublicKey() 返回的是 Null。
if (processAssemblyName != null &&
CompareBytes(currentAssemblyName.GetPublicKey(),
processAssemblyName.GetPublicKey()))
{
runningProcess = process;
returnValue = true;
break;
}
}
}
return returnValue;
}
/// 比较两个字节数组是否相等
private static bool CompareBytes(byte[] bytes1, byte[] bytes2)
{
if (bytes1 == null || bytes2 == null)
return false;
if (bytes1.Length != bytes2.Length)
return false;
for (int i = 0; i < bytes1.Length; i++)
{
if (bytes1[i] != bytes2[i])
return false;
}
return true;
}
特点:1) .exe 文件改名或路径改变,此方法可正常工作
2)需要对程序集进行签名。存在问题:速度稍慢
5 利用全局原子判断是否有相同的实例在运行
{
bool returnValue = true;
//没找到原子atom_test,则进程尚未运行,添加该原子
if (GlobalFindAtom("atom_test") == 0)
{
GlobalAddAtom("atom_test");
returnValue = false;
}
return returnValue;
}
[System.Runtime.InteropServices.DllImport("kernel32.dll")]
public static extern UInt32 GlobalAddAtom(String lpString); //添加原子
[System.Runtime.InteropServices.DllImport("kernel32.dll")]
public static extern UInt32 GlobalFindAtom(String lpString); //查找原子
[System.Runtime.InteropServices.DllImport("kernel32.dll")]
public static extern UInt32 GlobalDeleteAtom(UInt32 nAtom); //删除原子
特点:退出程序时要记得释放添加的原子,不然要到关机才会释放。
二 将指定进程的窗口设置为活动窗口
1 如果窗口最小化,设置为还原窗口;否则,直接将窗口设置为活动窗口,显示到前台
private static void SetForegroundProcess(Process process)
{
bool isIcon = IsIconic(process.MainWindowHandle);
// 窗口是否已最小化
if (isIcon)
{
// 还原窗口
ShowWindowAsync(process.MainWindowHandle, SW_RESTORE);
}
else
{
//如果期望窗口显示为Normal模式,可先做如下设置
//ShowWindowAsync(process.MainWindowHandle, SW_SHOWNORMAL);
// 将窗口设为前台窗口
SetForegroundWindow(process.MainWindowHandle);
}
}
2 根据窗口句柄控制窗口显示的相关Windows API 函数
/// <param name="hWnd">窗口句柄</param>
/// <returns>非零表示成功,零表示失败</returns>
[System.Runtime.InteropServices.DllImport("User32.dll")]
private static extern bool OpenIcon(IntPtr hWnd);
/// 窗口是否已最小化
[System.Runtime.InteropServices.DllImport("User32.dll")]
private static extern bool IsIconic(IntPtr hWnd);
/// 将窗口设为系统的前台窗口
[System.Runtime.InteropServices.DllImport("User32.dll")]
private static extern int SetForegroundWindow(IntPtr hWnd);
/// 与ShowWindow相似,只是这时的ShowWindow命令会投递到指定的窗口,然后进行异步处理。
/// 这样一来,就可控制从属于另一个进程的窗口的可视情况。
/// 同时无须担心另一个进程挂起的时候,自己的应用程序也会牵连其中返回值
/// <param name="cmdShow">为窗口指定可视性方面的一个命令</param>
/// <returns>如窗口之前是可见的,则返回TRUE(非零),否则返回FALSE(零)</returns>
[System.Runtime.InteropServices.DllImport("User32.dll")]
private static extern bool ShowWindowAsync(IntPtr hWnd, int cmdShow);
//相关常量
private const int SW_HIDE = 0; //隐藏窗口,活动状态给令一个窗口
private const int SW_SHOWNORMAL = 1; //用原来的大小和位置显示一个窗口,同时令其进入活动状态
private const int SW_SHOWMINIMIZED = 2; //最小化窗口,并将其激活
private const int SW_SHOWMAXIMIZED = 3; //最大化窗口,并将其激活
private const int SW_SHOWNOACTIVATE = 4; //用最近的大小和位置显示一个窗口,同时不改变活动窗口
private const int SW_RESTORE = 9; //用原来的大小和位置显示一个窗口,同时令其进入活动状态
private const int SW_SHOWDEFAULT = 10; //根据默认 创建窗口时的样式 来显示
#endregion
三 实现以下功能的代码示例
1) 程序只能运行一个实例。
2) 如果程序已经存在,且最小化,则还原那个程序。
3) 如果程序已经存在,且不是最小化(最大化或正常状态),则显示(注意:不是还原!)那个程序。
{
Process runningProcess = null;
bool isRunning = GetRunningProcessByProcessFullName(out runningProcess);
if (isRunning)
{
//已经有一个实例在运行
SetForegroundProcess(runningProcess);
}
else
{
//没有实例在运行
Application.Run(new Form1());
}
}
也可以先利用互斥量方法判断是否有相同的实例在运行,如果有,取该实例,其它处理方式如上。
其它思路:先用互斥法,如果程序已运行,则用sendmessage给程序发个消息,在程序中重写 WndProc方法,在此方法中处理消息。sendmessage中的句柄可以用System.Diagnostics.Process.GetProcessesByName(进程名)[0].MainWindowHandle获得。
C#让应用程序只运行一个实例的几种方法的更多相关文章
- 解决C#程序只允许运行一个实例的几种方法详解
解决C#程序只允许运行一个实例的几种方法详解 本篇文章是对C#中程序只允许运行一个实例的几种方法进行了详细的分析介绍,需要的朋友参考下 本文和大家讲一下如何使用C#来创建系统中只能有该程序的一个实例运 ...
- vc++高级班之窗口篇[4]---让程序只运行一个实例
大家都看过或者使用过类似只运行一个实例的程序,比如:QQ游戏.部分浏览器 等等! 让一个程序只运行一个实例的方法有多种,但是原理都类似,也就是在程序创建后,有窗口的程序在窗口创建前, 检查系统中是 ...
- VC 实现程序只运行一个实例,并激活已运行的程序
转载:http://blog.sina.com.cn/s/blog_4b44e1c00100bh69.html 进程的互斥运行:CreateMutex函数实现只运行一个程序实例 正常情况下,一个进程的 ...
- [VC]在VC++中实现让程序只运行一个实例的方法且实现该实例
方法一: 有时候在开发应用程序时,希望控制程序运行唯一的实例.例如,最常用的mp3播放软 件Winamp,由于它需要独占计算机中的音频设备,因此该程序只允许自身运行唯一的一个例程.在Visual C+ ...
- Java程序只运行一个实例[转]
如果希望你的Java程序只能存在一个实例,可以参考下面的用法. 原文链接:http://blog.csdn.net/yaerfeng/article/details/7264729 Java没有提供这 ...
- VC程序只运行一个实例,并在打开多个时激活原窗口
(一)单文档应用程序 1.在应用程序类C~~App::InitInstance()函数中判断是否已有一个应用程序实例正在运行 BOOL C~~App::InitInstance() { . ...
- 【转】delphi程序只允许运行一个实例的三种方法:
一. 创建互斥对象 在工程project1.dpr中创建互斥对象 Program project1 Uses Windows,Form, FrmMain in 'FrmMain.pas' ...
- Winform(C#)限制程序只运行一个实例
C#控制只运行开启一个程序 在这个例子中不需要调用ReleaseMutex,mutex会在程序结束时自动释放.为了防止mutex过早释放,在程序的最后调用下GC.KeepAlive (mutex). ...
- 让程序只运行一个实例(Delphi篇)(三种方法,其中使用全局原子的方法比较有意思)
Windows 下一个典型的特征就是多任务,我们可以同时打开多个窗口进行操作,也可以同时运行程序的多个实例,比如可以打开许多个资源管理器进行文件的移动复制操作.但有时出于某种考虑(比如安全性),我们要 ...
随机推荐
- develop brew app from here
https://brewx.qualcomm.com/brew/sdk/download.jsp?page=dx/en/brew31/ad/tl/overview the email is silen ...
- CentOS7.1 Liberty云平台之Dashboard篇(7)
控制节点: 一.安装及配置Dashboard 1.安装dashboard相关包 yum install openstack-dashboard 2.配置/etc/openstack-dashboard ...
- 超全面的JavaWeb笔记day03<JS对象&函数>
1.js的String对象(****) 2.js的Array对象 (****) 3.js的Date对象 (****) 获取当前的月 0-11,想要得到准确的月 +1 获取星期时候,星期日是 0 4.j ...
- Java精选笔记_集合【Map(映射)接口】
Map(映射)接口 简介 该集合存储键值对,一对一对的往里存,并且键是唯一的.要保证map集合中键的唯一性. 从Map集合中访问元素时,只要指定了Key,就能找到对应的Value. 关键字是以后用于检 ...
- Python 处理输入输出
sys.stdin.read() 用于接收标准输入,也就是让用户通过键盘进行输入sys.stdout.write() 用于打印标准输出,也就是把输入的数据输出到屏幕sys.stderr.write() ...
- chmod 777 修改权限
http://william71.blogbus.com/logs/33484772.html 在Unix和Linux的各种操作系统下,每个文件(文件夹也被看作是文件)都按读.写.运行设定权限.例如我 ...
- lua垃圾回收机制
一.检测lua内存泄漏: 注:使用“collectgarbage("collect")”,局部变量v被回收,my_list没有被回收. 注:局部变量v占用的内存被回收. 注:将my ...
- salt-stack更换主机名
author:headsen chen date: 2018-09-30 11:22:40 1,建立master端和client端的正常连接 #master yum -y install epel ...
- [css]演示:纯CSS实现的右侧底部简洁悬浮效果
<!DOCTYPE HTML> <html> <head> <meta charset="utf-8"> <meta name ...
- 暴力破解工具hydra
Hydra是一个并行登录的裂解装置,它支持众多的协议来攻击.新的模块很容易的添加,旁边,它是灵活的,而且速度非常快. 首先安装的是hydra的支持库包软件. yum -y install openss ...