Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 669|回复: 0

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

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-4-10 15:17:16 | 显示全部楼层 |阅读模式

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

     {

      //不存在

     }

     

     

     

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-12-22 17:29 , Processed in 0.057722 second(s), 28 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表