要实现程序的互斥,通常有下面几种方式,下面用 C# 语言来实现:

方法一:
使用线程互斥变量. 通过定义互斥变量来判断是否已运行实例.
把program.cs文件里的Main()函数改为如下代码:

using System;

using System.Windows.Forms;

using System.Runtime.InteropServices;

namespace NetTools

{

static class Program

{

[DllImport("user32.dll")]

private static extern bool
FlashWindow(IntPtr hWnd, bool bInvert);

[DllImport("user32.dll")]

private
static extern bool FlashWindowEx(int
pfwi);

/// <summary>

/// 应用程序的主入口点。

/// </summary>

[STAThread]

static void Main()

{

bool
runone;

System.Threading.Mutex run = new
System.Threading.Mutex(true, "single_test",
out runone);

if
(runone)

{

run.ReleaseMutex();

Application.EnableVisualStyles();

Application.SetCompatibleTextRenderingDefault(false);

FrmRemote frm = new FrmRemote();

int
hdc = frm.Handle.ToInt32(); // write to ...

Application.Run(frm);

IntPtr
a = new IntPtr(hdc);

}

else

{

MessageBox.Show("已经运行了一个实例了。");

//IntPtr
hdc = new IntPtr(1312810); // read from...

//bool
flash = FlashWindow(hdc, true);

}

}

}

}

说明:程序中通过语句 System.Threading.Mutex run =
new System.Threading.Mutex(true, "single_test", out runone);来创建一个互斥体变量run,其中"single_test"为互斥体名,在此方法返回时,如果创建了局部互斥体或指定的命名系统互斥体,则布尔值runone为true;如果指定的命名系统互斥体已存在,则为 false。已命名的互斥体是系统范围的。

方法二:采用判断进程的方式,我们在运行程序前,查找进程中是否有同名的进程,同时运行位置也相同程,如是没有运行该程序,如果有就就不运行.在C#中应用System.Diagnostics名字空间中的Process类来实现,主要代码如下: 
1,在program.cs文件中添加函数如下:

public static System.Diagnostics.Process
RunningInstance()

{

System.Diagnostics.Process current = System.Diagnostics.Process.GetCurrentProcess();

System.Diagnostics.Process[] processes = System.Diagnostics.Process.GetProcesses();

foreach
(System.Diagnostics.Process process in processes) //查找相同名称的进程

{

if
(process.Id != current.Id) //忽略当前进程

{ //确认相同进程的程序运行位置是否一样.

if
(System.Reflection.Assembly.GetExecutingAssembly().Location.Replace("/", @"/")
== current.MainModule.FileName)

{ //Return
the other process instance.

return process;

}

}

} //No
other instance was found, return null.

return
null;

}

2,把Main ()函数改为如下代码:

static void Main()

{

if
(RunningInstance() == null)

{

Application.EnableVisualStyles();

Application.SetCompatibleTextRenderingDefault(false);

Application.Run(new Form1());

}

else

{

MessageBox.Show("已经运行了一个实例了。");

}

}

方法三:全局原子法,创建程序前,先检查全局原子表中看是否存在特定原子A(创建时添加的),存在时停止创建,说明该程序已运行了一个实例;不存在则运行程序并想全局原子表中添加特定原子A;退出程序时要记得释放特定的原子A哦,不然要到关机才会释放。C#实现如下: 
1.申明WinAPI函数接口

[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); //删除原子

2.修改Main()函数如下:

static void Main()

{

if
(GlobalFindAtom("jiaao_test") ==
77856768) //没找到原子"jiaao_test"

{

GlobalAddAtom("jiaao_test"); //添加原子"jiaao_test"

Application.EnableVisualStyles();

Application.SetCompatibleTextRenderingDefault(false);

Application.Run(new Form1());

}

else

{

MessageBox.Show("已经运行了一个实例了。");

}

}

3.FormClosed事件中添加如下代码: 
      
GlobalDeleteAtom(GlobalFindAtom("jiaao_test"));//删除原子"jiaao_test"

方法四:通过进程判断是否启动:

static class Program

{

/// <summary>

/// 应用程序的主入口点。

/// </summary>

[STAThread]

static void Main()

{

//获取当前进程的ID

int pId = Process.GetCurrentProcess().Id;

bool isRun = false;

foreach (Process p in Process.GetProcessesByName("CallMaster"))

{

//取得当前程序的进程,进行比较

if (Common.GetPath().ToLower() == p.MainModule.FileName.ToLower())

{

if (pId != p.Id)

{

isRun = true;

break;

}

}

}

if (isRun==true)

{

Application.Exit();

return;

}

Application.EnableVisualStyles();

Application.SetCompatibleTextRenderingDefault(false);

Application.Run(new frmMain());

}

}

利用放射获取当前应用程序的全路径:

public static string GetPath()

{

return System.Reflection.Assembly.GetExecutingAssembly().Location;

}

方法五:通过线程互斥判断是否启动:

static class Program

{

private static System.Threading.Mutex mutex;

/// <summary>

/// 应用程序的主入口点。

/// </summary>

[STAThread]

static void Main()

{

Application.EnableVisualStyles();

Application.SetCompatibleTextRenderingDefault(false);

mutex = new System.Threading.Mutex(true, "OnlyRun");

if (mutex.WaitOne(0, false))

{

Application.Run(new MainForm());

}

else

{

MessageBox.Show("程序已经在运行!", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);

Application.Exit();

}

}

}

另附:c#中怎样判断一个程序是否正在运行?

if (System.Diagnostics.Process.GetProcessesByName("程序进程中的名称").ToList().Count > 0)

{

//存在

}

else

{

//不存在

}

解决C# WINFORM程序只允许运行一个实例的几种方法详解的更多相关文章

  1. 解决C#程序只允许运行一个实例的几种方法详解

    解决C#程序只允许运行一个实例的几种方法详解 本篇文章是对C#中程序只允许运行一个实例的几种方法进行了详细的分析介绍,需要的朋友参考下 本文和大家讲一下如何使用C#来创建系统中只能有该程序的一个实例运 ...

  2. 【转】delphi程序只允许运行一个实例的三种方法:

    一.        创建互斥对象 在工程project1.dpr中创建互斥对象 Program project1 Uses Windows,Form, FrmMain in 'FrmMain.pas' ...

  3. C#程序只允许运行一个实例的解决方案

    最近在做winform的程序中,需要只能打开一个程序,如果已经存在,则激活该程序的窗口,并显示在最前端.在网上google了一哈,找到了很多的解决方案.这里我整理了3种方案,并经过了测试,现和朋友们分 ...

  4. C#让应用程序只运行一个实例的几种方法

    一 判断是否有相同的实例已经运行 1 根据“Mutex”判断是否有相同的实例在运行 /// <returns>已有实例运行返回true,否则为false</returns>pu ...

  5. Winform程序只允许运行一个程序实例

    /// <summary> /// 应用程序的主入口点. /// </summary> [STAThread] static void Main() { Application ...

  6. python 单例模式,一个类只能生成唯一的一个实例,重写__new__方法详解

    单例:一个类只能生成唯一的一个实例 每个类只要被实例化了,他的私有属性 '_instance'就会被赋值,这样理解对吗 对 #方法1,实现__new__方法 #并在将一个类的实例绑定到类变量_inst ...

  7. WinForm判断程序是否已经在运行,且只允许运行一个实例

    我们开发WinFrom程序,很多时候都希望程序只有一个实例在运行,避免运行多个同样的程序,一是没有意义,二是容易出错. 为了更便于使用,笔者整理了一段自己用的代码,可以判断程序是否在运行,只运行一个实 ...

  8. C# WinForm中如何让当前应用程序只允许启动一个实例

    我们在WinForm开发中,很多情况下是需要只允许让用户运行一个实例,那么代码其实很简单.只需要修改Program.cs文件,代码如下 static class Program { /// <s ...

  9. 转 C# 只允许运行一个实例

    来源:http://blog.csdn.net/jin20000/article/details/3136791 互斥进程(程序), 简单点说,就是在系统中只能有该程序的一个实例运行. 现在很多软件都 ...

随机推荐

  1. CDC之CreateCompatibleDC与BitBlt

    CreateCompatibleDC 创建一个与指定设备一致的内存设备描写叙述表. HDC CreateCompatibleDC(HDC hdc //设备描写叙述表句柄); 參数 hdc 现有的设备描 ...

  2. source insight完全卸载

    由于不知名原因 source insight崩溃了,使用自带的卸载,完成之后重新安装软件注册还是出问题.在网上搜索资料发现就是删除注册表中的内容. 由于列出的删除项目不完全,导致还是出问题. 最后删除 ...

  3. 理解JavaScript中的事件流

    原文地址:http://my.oschina.net/sevenhdu/blog/332014 目录[-] 事件冒泡 事件捕获 DOM事件流 当浏览器发展到第四代时(IE4和Netscape Comm ...

  4. TurboLinux11system»adjtimex简介

    Adjtimex介绍 linux 系统有两个时钟:一个是由主板电池驱动的“Real Time Clock”也叫做RTC或者叫CMOS时钟,硬件时钟.当操作系统关机的时候,用这个来记录时间,但是对于运行 ...

  5. Cohen-SutherLand算法(编码算法)

    转自:http://my.oschina.net/liqiong/blog/4921 Cohen-SutherLand算法(编码算法)   基本思想:对于每条线段P1P2,分为三种情况处理: (1)若 ...

  6. Hadoop Combiners

    In the last post and in the preceding one we saw how to write a MapReduce program for finding the to ...

  7. docker和虚拟化技术的区别

    1.docker和虚拟化技术的区别 Docker 扩展了 Linux 容器(Linux Containers),或着说 LXC,通过一个高层次的 API 为进程单独提供了一个轻量级的虚拟环境.Dock ...

  8. Java多线程学习(吐血超具体总结)

    林炳文Evankaka原创作品. 转载请注明出处http://blog.csdn.net/evankaka 写在前面的话:此文仅仅能说是java多线程的一个入门.事实上Java里头线程全然能够写一本书 ...

  9. linux系统安装gcc

    安装nginx时会遇到错误提示gcc: command not found,需要先安装gcc 在centos7上安装成功. # yum group list # yum group install & ...

  10. HTML:几个常见的列表标签

    介绍: 在网页中列表是很常见的标签,主要分为有序标签.无序标签.列表嵌套.定义标签 有序标签:<ol><li></li><ol> 无序标签:<ul ...