1. 程式人生 > >只開啟一個程式,如果第二次開啟則自動將第一個程式顯示到桌面

只開啟一個程式,如果第二次開啟則自動將第一個程式顯示到桌面

using System;
using System.Collections.Generic;
using System.Windows.Forms;
using System.Runtime.InteropServices;
using System.Diagnostics;
using System.Reflection;


namespace AppLed
{
    static class Program
    {
        //防止程式執行多個例項的方法有多種,如:通過使用互斥量和程序名等.而我想要實現的是:在程式執行多個例項時啟用的是第一個例項,使其獲得焦點,並在前端顯示.
        //主要用到兩個API 函式:
        //ShowWindowAsync 該函式設定由不同執行緒產生的視窗的顯示狀態。
        //SetForegroundWindow 該函式將建立指定視窗的執行緒設定到前臺,並且啟用該視窗。鍵盤輸入轉向該視窗,併為使用者改各種可視的記號。系統給建立前臺視窗的執行緒分配的許可權稍高於其他執行緒。

        /// <summary>
        /// 該函式設定由不同執行緒產生的視窗的顯示狀態。
        /// </summary>
        /// <param name="hWnd">視窗控制代碼</param>
        /// <param name="cmdShow">指定視窗如何顯示。檢視允許值列表,請查閱ShowWlndow函式的說明部分。</param>
        /// <returns>如果函式原來可見,返回值為非零;如果函式原來被隱藏,返回值為零。</returns>
        [DllImport("User32.dll")]
        private static extern bool ShowWindowAsync(IntPtr hWnd, int cmdShow);
        /// <summary>
        /// 該函式將建立指定視窗的執行緒設定到前臺,並且啟用該視窗。鍵盤輸入轉向該視窗,併為使用者改各種可視的記號。系統給建立前臺視窗的執行緒分配的許可權稍高於其他執行緒。
        /// </summary>
        /// <param name="hWnd">將被啟用並被調入前臺的視窗控制代碼。</param>
        /// <returns>如果視窗設入了前臺,返回值為非零;如果視窗未被設入前臺,返回值為零。</returns>
        [DllImport("User32.dll")]
        private static extern bool SetForegroundWindow(IntPtr hWnd);
        private const int WS_SHOWNORMAL = 1;


        /// <summary>
        /// 應用程式的主入口點。
        /// </summary>
        [STAThread]
        static void Main()
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(false);
            Process instance = RunningInstance();
            if (instance == null)
            {
                Application.Run(new FrmLed());
            }
            else
            {
                HandleRunningInstance(instance);
            }
        }
        /// <summary>
        /// 獲取正在執行的例項,沒有執行的例項返回null;
        /// </summary>
        public static Process RunningInstance()
        {
            Process current = Process.GetCurrentProcess();
            Process[] processes = Process.GetProcessesByName(current.ProcessName);
            foreach (Process process in processes)
            {
                if (process.Id != current.Id)
                {
                    if (Assembly.GetExecutingAssembly().Location.Replace("/", "\\") == current.MainModule.FileName)
                    {
                        return process;
                    }
                }
            }
            return null;
        }

        /// <summary>
        /// 顯示已執行的程式。
        /// </summary>
        public static void HandleRunningInstance(Process instance)
        {
            //MessageBox.Show("ID:"+instance.Id .ToString()+"--控制代碼"+instance.MainWindowHandle.ToString() + "--正常視窗" + WS_SHOWNORMAL + "--" + ShowWindowAsync(instance.MainWindowHandle, WS_SHOWNORMAL) + "--" + SetForegroundWindow(instance.MainWindowHandle));
            ShowWindowAsync(instance.MainWindowHandle, WS_SHOWNORMAL); //顯示,可以註釋掉
            SetForegroundWindow(instance.MainWindowHandle);            //放到前端
        }
    }
}

 2種方法

#region  確保程式只執行一個例項
        private static Process RunningInstance()
        {
            Process current = Process.GetCurrentProcess();
            Process[] processes = Process.GetProcessesByName(current.ProcessName);
            //遍歷與當前程序名稱相同的程序列表  
            foreach (Process process in processes)
            {
                //如果例項已經存在則忽略當前程序  
                if (process.Id != current.Id)
                {
                    //保證要開啟的程序同已經存在的程序來自同一檔案路徑
                    if (Assembly.GetExecutingAssembly().Location.Replace("/", "\\") == current.MainModule.FileName)
                    {
                        //返回已經存在的程序
                        return process;
                    }
                }
            }
            return null;
        }

        private static void HandleRunningInstance(Process instance)
        {
            MessageBox.Show("監控系統已經在執行!", "提示資訊", MessageBoxButtons.OK, MessageBoxIcon.Information);
            ShowWindowAsync(instance.MainWindowHandle, 1);  //呼叫api函式,正常顯示視窗
            SetForegroundWindow(instance.MainWindowHandle); //將視窗放置最前端
        }
        [DllImport("User32.dll")]
        private static extern bool ShowWindowAsync(System.IntPtr hWnd, int cmdShow);
        [DllImport("User32.dll")]
        private static extern bool SetForegroundWindow(System.IntPtr hWnd);
        #endregion

 

只有視窗最小化的時候可以達到此效果,如果隱藏到托盤則無法將開啟的程式顯示到桌面

using System;
using System.Collections.Generic;
using System.Windows.Forms;
using System.Runtime.InteropServices;
using System.Diagnostics;
using System.Reflection;


namespace AppLed
{
    static class Program
    {
        //防止程式執行多個例項的方法有多種,如:通過使用互斥量和程序名等.而我想要實現的是:在程式執行多個例項時啟用的是第一個例項,使其獲得焦點,並在前端顯示.
        //主要用到兩個API 函式:
        //ShowWindowAsync 該函式設定由不同執行緒產生的視窗的顯示狀態。
        //SetForegroundWindow 該函式將建立指定視窗的執行緒設定到前臺,並且啟用該視窗。鍵盤輸入轉向該視窗,併為使用者改各種可視的記號。系統給建立前臺視窗的執行緒分配的許可權稍高於其他執行緒。

        /// <summary>
        /// 該函式設定由不同執行緒產生的視窗的顯示狀態。
        /// </summary>
        /// <param name="hWnd">視窗控制代碼</param>
        /// <param name="cmdShow">指定視窗如何顯示。檢視允許值列表,請查閱ShowWlndow函式的說明部分。</param>
        /// <returns>如果函式原來可見,返回值為非零;如果函式原來被隱藏,返回值為零。</returns>
        [DllImport("User32.dll")]
        private static extern bool ShowWindowAsync(IntPtr hWnd, int cmdShow);
        /// <summary>
        /// 該函式將建立指定視窗的執行緒設定到前臺,並且啟用該視窗。鍵盤輸入轉向該視窗,併為使用者改各種可視的記號。系統給建立前臺視窗的執行緒分配的許可權稍高於其他執行緒。
        /// </summary>
        /// <param name="hWnd">將被啟用並被調入前臺的視窗控制代碼。</param>
        /// <returns>如果視窗設入了前臺,返回值為非零;如果視窗未被設入前臺,返回值為零。</returns>
        [DllImport("User32.dll")]
        private static extern bool SetForegroundWindow(IntPtr hWnd);
        private const int WS_SHOWNORMAL = 1;


        /// <summary>
        /// 應用程式的主入口點。
        /// </summary>
        [STAThread]
        static void Main()
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(false);
            Process instance = RunningInstance();
            if (instance == null)
            {
                Application.Run(new FrmLed());
            }
            else
            {
                HandleRunningInstance(instance);
            }
        }
        /// <summary>
        /// 獲取正在執行的例項,沒有執行的例項返回null;
        /// </summary>
        public static Process RunningInstance()
        {
            Process current = Process.GetCurrentProcess();
            Process[] processes = Process.GetProcessesByName(current.ProcessName);
            foreach (Process process in processes)
            {
                if (process.Id != current.Id)
                {
                    if (Assembly.GetExecutingAssembly().Location.Replace("/", "\\") == current.MainModule.FileName)
                    {
                        return process;
                    }
                }
            }
            return null;
        }

        /// <summary>
        /// 顯示已執行的程式。
        /// </summary>
        public static void HandleRunningInstance(Process instance)
        {
            //MessageBox.Show("ID:"+instance.Id .ToString()+"--控制代碼"+instance.MainWindowHandle.ToString() + "--正常視窗" + WS_SHOWNORMAL + "--" + ShowWindowAsync(instance.MainWindowHandle, WS_SHOWNORMAL) + "--" + SetForegroundWindow(instance.MainWindowHandle));
            ShowWindowAsync(instance.MainWindowHandle, WS_SHOWNORMAL); //顯示,可以註釋掉
            SetForegroundWindow(instance.MainWindowHandle);            //放到前端
        }
    }
}

 2種方法

#region  確保程式只執行一個例項
        private static Process RunningInstance()
        {
            Process current = Process.GetCurrentProcess();
            Process[] processes = Process.GetProcessesByName(current.ProcessName);
            //遍歷與當前程序名稱相同的程序列表  
            foreach (Process process in processes)
            {
                //如果例項已經存在則忽略當前程序  
                if (process.Id != current.Id)
                {
                    //保證要開啟的程序同已經存在的程序來自同一檔案路徑
                    if (Assembly.GetExecutingAssembly().Location.Replace("/", "\\") == current.MainModule.FileName)
                    {
                        //返回已經存在的程序
                        return process;
                    }
                }
            }
            return null;
        }

        private static void HandleRunningInstance(Process instance)
        {
            MessageBox.Show("監控系統已經在執行!", "提示資訊", MessageBoxButtons.OK, MessageBoxIcon.Information);
            ShowWindowAsync(instance.MainWindowHandle, 1);  //呼叫api函式,正常顯示視窗
            SetForegroundWindow(instance.MainWindowHandle); //將視窗放置最前端
        }
        [DllImport("User32.dll")]
        private static extern bool ShowWindowAsync(System.IntPtr hWnd, int cmdShow);
        [DllImport("User32.dll")]
        private static extern bool SetForegroundWindow(System.IntPtr hWnd);
        #endregion

 

只有視窗最小化的時候可以達到此效果,如果隱藏到托盤則無法將開啟的程式顯示到桌面