要实现程序的互斥,通常有下面几种方式,下面用 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. HDU 4686 Arc of Dream (2013多校9 1001 题,矩阵)

    Arc of Dream Time Limit: 2000/2000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others)Tota ...

  2. Get just enough boost voltage - current-mirror circuit - VOUT tracks VIN varies

    Adding a current-mirror circuit to a typical boost circuit allows you to select the amount of boost ...

  3. Druid如何自动根据URL自动识别DriverClass的

    Druid是根据url前缀来识别DriverClass的,这样使得配置更方便简洁. 前缀 DriverCLass 描述信息 jdbc:odps com.aliyun.odps.jdbc.OdpsDri ...

  4. C#中有关资源、BeginInvoke, Invoke和事件的事情

    事情是这么来的,我开发的一个程序报了一个错误 “在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke错误”. 然后我在网上查资料,发现一个有意思的问题,文章出处为“在创建窗口 ...

  5. 两步改动CentOS主机名称

    在CentOS系统中,改动主机名称的过程例如以下: 1. 改动network文件 编辑network文件.配置例如以下: vi /etc/sysconfig/network 配置 NETWORKING ...

  6. 机器学习: 神经网络中的Error函数

    利用神经网络做分类的时候,可以将神经网络看成一个mapping function,从输入到输出经过复杂的非线性变换.而输出的预测值与实际的目标值总是存在一定偏差的,一般利用这种偏差建立error 函数 ...

  7. 网页采集(通过HtmlAgilityPack+XPath)

    有HtmlAgilityPack这个类库可以更方便地对HTML内容进行分析和提取.因此今天特别学习和实践了一下HtmlAgilityPack和XPath,并作下笔记. 1.下载HtmlAgilityP ...

  8. GIF 屏幕录制工具

    LICEcap是一款屏幕录制工具,支持导出 GIF 动绘图片格式,轻量级.使用简单,录制过程中能够任意改变录屏范围.             .捕捉屏幕的区域并保存为.GIF动画(便于网络公布)或.L ...

  9. 使用 Kafka 和 Spark Streaming 构建实时数据处理系统(转)

    原文链接:http://www.ibm.com/developerworks/cn/opensource/os-cn-spark-practice2/index.html?ca=drs-&ut ...

  10. [转]nginx折腾记(HTTP性能能测试,与Apache对比)

      话说nginx在大压力的环境中比apache的表现要好,于是下载了一个来折腾一下. 下载并编译安装,我的编译过程有点特别: 1.去除调试信息,修改$nginx_setup_path/auto/cc ...