原文:DevExpress Winform使用单例运行程序方法和非DevExpress使用Mutex实现程序单实例运行且运行则激活窗体的方法

版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。

网上关于C#单例运行程序的方法都是比较简单,有些甚至是无法实现功能的,不知道他们试没试过就发帖,因为自己之前都是用第三方控件DevExpress,单例运行也是用它本身自带的一个方法,调用此方法需要引用DevExpress的DevExpress.DevAV.v17.1.Data.dll


  1. static void Main()
  2. {
  3. var appName= Process.GetCurrentProcess().ProcessName;
  4. using (DevExpress.Internal.DevAVDataDirectoryHelper.SingleInstanceApplicationGuard(appName, out exit))
  5. {
  6. if (exit)
  7. return;
  8. Application.EnableVisualStyles();
  9. Application.SetCompatibleTextRenderingDefault(false);
  10. Application.Run(new Form1());
  11. }
  12. }

以前懒,看到有现成的就拿现成的,也没去想具体如何实现,刚好有人问起,才决定去看看源码,发现DevExpress的这个方法原来是用了Mutex来实现的,整合了一下资料,重新写一个供C#桌面应用程序通用的单例运行方法,代码如下:

其中需要引用以下命名空间:


  1. using System;
  2. using System.Diagnostics;
  3. using System.Drawing;
  4. using System.Runtime.InteropServices;
  5. using System.Security;
  6. using System.Threading;

  1. /// <summary>
  2. /// 单实例应用程序--by涛神
  3. /// </summary>
  4. public class SingleInstanceApplication
  5. {
  6. /// <summary>
  7. /// 判断应用是否运行
  8. /// </summary>
  9. /// <param name="processName"></param>
  10. /// <param name="exit"></param>
  11. /// <returns></returns>
  12. public static IDisposable Guard(out bool exit)
  13. {
  14. Process currentProcess = Process.GetCurrentProcess();
  15. var processName = currentProcess.ProcessName;
  16. Mutex mutex = new Mutex(true, processName, out bool createNew);
  17. if (createNew)
  18. {
  19. exit = false;
  20. }
  21. else
  22. {
  23. Process[] processesByName = Process.GetProcessesByName(currentProcess.ProcessName);
  24. int num = 0;
  25. while (num < (int)processesByName.Length)
  26. {
  27. Process process = processesByName[num];
  28. if (process.Id == currentProcess.Id || !(process.MainWindowHandle != IntPtr.Zero))
  29. {
  30. num++;
  31. }
  32. else
  33. {
  34. WinApiHelper.SetForegroundWindow(process.MainWindowHandle);
  35. WinApiHelper.RestoreWindowAsync(process.MainWindowHandle);
  36. break;
  37. }
  38. }
  39. exit = true;
  40. }
  41. return mutex;
  42. }
  43. private static class WinApiHelper
  44. {
  45. [SecuritySafeCritical]
  46. public static bool IsMaxmimized(IntPtr hwnd)
  47. {
  48. WinApiHelper.Import.WINDOWPLACEMENT wINDOWPLACEMENT = new WinApiHelper.Import.WINDOWPLACEMENT();
  49. if (!WinApiHelper.Import.GetWindowPlacement(hwnd, ref wINDOWPLACEMENT))
  50. {
  51. return false;
  52. }
  53. return wINDOWPLACEMENT.showCmd == WinApiHelper.Import.ShowWindowCommands.ShowMaximized;
  54. }
  55. [SecuritySafeCritical]
  56. public static bool RestoreWindowAsync(IntPtr hwnd)
  57. {
  58. return WinApiHelper.Import.ShowWindowAsync(hwnd, (WinApiHelper.IsMaxmimized(hwnd) ? 3 : 9));
  59. }
  60. [SecuritySafeCritical]
  61. public static bool SetForegroundWindow(IntPtr hwnd)
  62. {
  63. return WinApiHelper.Import.SetForegroundWindow(hwnd);
  64. }
  65. private static class Import
  66. {
  67. [DllImport("user32.dll", CharSet = CharSet.None, ExactSpelling = false)]
  68. public static extern bool GetWindowPlacement(IntPtr hWnd, ref WinApiHelper.Import.WINDOWPLACEMENT lpwndpl);
  69. [DllImport("user32.dll", CharSet = CharSet.None, ExactSpelling = false)]
  70. public static extern bool SetForegroundWindow(IntPtr hWnd);
  71. [DllImport("user32.dll", CharSet = CharSet.None, ExactSpelling = false)]
  72. public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
  73. public enum ShowWindowCommands
  74. {
  75. Hide,
  76. Normal,
  77. ShowMinimized,
  78. ShowMaximized,
  79. ShowNoActivate,
  80. Show,
  81. Minimize,
  82. ShowMinNoActive,
  83. ShowNA,
  84. Restore,
  85. ShowDefault,
  86. ForceMinimize
  87. }
  88. public struct WINDOWPLACEMENT
  89. {
  90. public int length;
  91. public int flags;
  92. public WinApiHelper.Import.ShowWindowCommands showCmd;
  93. public Point ptMinPosition;
  94. public Point ptMaxPosition;
  95. public Rectangle rcNormalPosition;
  96. }
  97. }
  98. }
  99. }

调用的demo如下,亲测可以,若无法实现的请留言讨论(本文章是原创文章,转载请标明,盗版必究,觉得文章对你帮助的可以点个赞表示支持一下,谢谢):


  1. static void Main()
  2. {
  3. using(SingleInstanceApplication.Guard(out bool exit))
  4. {
  5. if (exit)
  6. return;
  7. Application.EnableVisualStyles();
  8. Application.SetCompatibleTextRenderingDefault(false);
  9. Application.Run(new Form1());
  10. }
  11. }

DevExpress Winform使用单例运行程序方法和非DevExpress使用Mutex实现程序单实例运行且运行则激活窗体的方法的更多相关文章

  1. java——多线程——单例模式的static方法和非static方法是否是线程安全的?

    单例模式的static方法和非static方法是否是线程安全的? 答案是:单例模式的static方法和非static方法是否是线程安全的,与单例模式无关.也就说,如果static方法或者非static ...

  2. synchronized 修饰在 static方法和非static方法的区别

    Java中synchronized用在静态方法和非静态方法上面的区别 在Java中,synchronized是用来表示同步的,我们可以synchronized来修饰一个方法.也可以synchroniz ...

  3. Java中synchronized 修饰在static方法和非static方法的区别

    [问题描述]关于Java中synchronized 用在实例方法和对象方法上面的区别 [问题分析]大家都知道,在Java中,synchronized 是用来表示同步的,我们可以synchronized ...

  4. 关于C#单例Singleton的看法和使用

    首先明白一点,什么是单例模式? 单例模式是指一个类在一个应用程序运行时仅仅实例化一次,以后所有的调用都使用第一次实例化的对象,是应用程序级别的,与session,用户等无关,它比全局参数或静态类方式更 ...

  5. iOS——Swift开发中的单例设计模式(摘译,非原创)

    最近在开发一个小的应用,遇到了一些Objective-c上面常用的单例模式,但是swift上面还是有一定区别的,反复倒来倒去发现不能按常理(正常的oc to swift的方式)出牌,因此搜索了一些帖子 ...

  6. java static成员变量方法和非static成员变量方法的区别

    这里的普通方法和成员变量是指,非静态方法和非静态成员变量首先static是静态的意思,是修饰符,可以被用来修饰变量或者方法. static成员变量有全局变量的作用       非static成员变量则 ...

  7. 表单元素的submit()方法和onsubmit事件

    1.表单元素中出现了name="submit"的元素 2.elemForm.submit();不会触发表单的onsubmit事件 3.动态创建表单时遇到的问题 表单元素拥有subm ...

  8. 表单元素的submit()方法和onsubmit事件(转)

    1.表单元素中出现了name="submit"的元素 2.elemForm.submit();不会触发表单的onsubmit事件 3.动态创建表单时遇到的问题 表单元素拥有subm ...

  9. static方法和非static方法的区别

    ●生命周期(Lifecycle):静态方法(Static Method)与静态成员变量一样,属于类本身,在类装载的时候被装载到内存(Memory),不自动进行销毁,会一直存在于内存中,直到JVM关闭. ...

随机推荐

  1. git的HEAD指针操作

    学习操作HEAD指针,具体如下: - 查看Git版本信息 - 移动指针 - 通过移动HEAD指针恢复数据 - 合并版本 拓扑图:

  2. 脚本_获取本机 MAC 地址

    #!bin/bash#作者:liusingbon#功能:获取本机 MAC 地址ip a s |awk 'BEGIN{print "本机 MAC 地址信息如下:"}/^[0-9]/{ ...

  3. 【LeetCode】前缀树 trie(共14题)

    [208]Implement Trie (Prefix Tree) (2018年11月27日) 实现基本的 trie 树,包括 insert, search, startWith 操作等 api. 题 ...

  4. Linux 部署或升级openssh7.5p1

    运维Linux系统,部署或升级openssh是经常面临的事,以下已redhat6和redhat7为例. 在redhat6中部署openssh会有什么坑,在编辑openssh源码包时会报一些类似的错误, ...

  5. group_by

    1.按照一个列或者多个列对数据分组 2.对每个组进行聚合操作 3. 对聚合后的结果进行判断 1. select avg(score) as score from teacher 2. select   ...

  6. Codeforces Round #585 (Div. 2) [补题]

    前言 2019.9.16 昨天下午就看了看D题,没有写对,因为要补作业,快点下机了,这周争取把题补完. 2019.9.17 这篇文章或者其他文章难免有错别字不被察觉,请读者还是要根据意思来读,不要纠结 ...

  7. fiddler使用介绍

    Fiddler的详细介绍 Fiddler的详细介绍 一.Fiddler与其他抓包工具的区别 1.Firebug虽然可以抓包,但是对于分析http请求的详细信息,不够强大.模拟http请求的功能也不够, ...

  8. SpringBoot之Web进阶

    .. 另外包括Springboot常用技术整合  以及项目上的应用

  9. HTTP通信安全和Web攻击技术

    一.HTTPS,确保Web安全 在HTTP协议中可能存在信息窃听或身份伪装等安全问题,HTTP的不足: 通信使用明文(不加密),内容可能会被窃听  不验证通信方的身份,因此有可能遭遇伪装 无法证明报文 ...

  10. php上传文件的原理

    文件上传原理 将客户端的文件上传到服务器,再将服务器的临时文件上传到指定目录 客户端配置 提交表单 表单的发送方式为post 添加enctype="multipart/form-data&q ...