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

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

要实现程序的互斥,通常有下面几种方式,下面用 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

 {

  //不存在

 }

 

 

 

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
智慧校园整体解决方案是响应国家教育信息化政策,结合教育改革和技术创新的产物。该方案以物联网、大数据、人工智能和移动互联技术为基础,旨在打造一个安全、高效、互动且环保的教育环境。方案强调从数字化校园向智慧校园的转变,通过自动数据采集、智能分析和按需服务,实现校园业务的智能化管理。 方案的总体设计原则包括应用至上、分层设计和互联互通,确保系统能够满足不同用户角色的需求,并实现数据和资源的整合与共享。框架设计涵盖了校园安全、管理、教学、环境等多个方面,构建了一个全面的校园应用生态系统。这包括智慧安全系统、校园身份识别、智能排课及选课系统、智慧学习系统、精品录播教室方案等,以支持个性化学习和教学评估。 建设内容突出了智慧安全和智慧管理的重要性。智慧安全管理通过分布式录播系统和紧急预案一键启动功能,增强校园安全预警和事件响应能力。智慧管理系统则利用物联网技术,实现人员和设备的智能管理,提高校园运营效率。 智慧教学部分,方案提供了智慧学习系统和精品录播教室方案,支持专业级学习硬件和智能化网络管理,促进个性化学习和教学资源的高效利用。同时,教学质量评估中心和资源应用平台的建设,旨在提升教学评估的科学性和教育资源的共享性。 智慧环境建设则侧重于基于物联网的设备管理,通过智慧教室管理系统实现教室环境的智能控制和能效管理,打造绿色、节能的校园环境。电子班牌和校园信息发布系统的建设,将作为智慧校园的核心和入口,提供教务、一卡通、图书馆等系统的集成信息。 总体而言,智慧校园整体解决方案通过集成先进技术,不仅提升了校园的信息化水平,而且优化了教学和管理流程,为学生、教师和家长提供了更加便捷、个性化的教育体验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值