解決C#程序只允許運行一個實例的幾種方法詳解
更新時間:2013年05月20日 10:41:25 作者:
本篇文章是對C#中程序只允許運行一個實例的幾種方法進行了詳細的分析介紹,需要的朋友參考下
本文和大家講一下如何使用C#來創(chuàng)建系統(tǒng)中只能有該程序的一個實例運行。
要實現(xiàn)程序的互斥,通常有下面幾種方式,下面用 C# 語言來實現(xiàn):
方法一:
使用線程互斥變量. 通過定義互斥變量來判斷是否已運行實例.
把program.cs文件里的Main()函數(shù)改為如下代碼:
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);來創(chuàng)建一個互斥體變量run,其中"single_test"為互斥體名,在此方法返回時,如果創(chuàng)建了局部互斥體或指定的命名系統(tǒng)互斥體,則布爾值runone為true;如果指定的命名系統(tǒng)互斥體已存在,則為 false。已命名的互斥體是系統(tǒng)范圍的。
方法二:采用判斷進程的方式,我們在運行程序前,查找進程中是否有同名的進程,同時運行位置也相同程,如是沒有運行該程序,如果有就就不運行.在C#中應用System.Diagnostics名字空間中的Process類來實現(xiàn),主要代碼如下:
1,在program.cs文件中添加函數(shù)如下:
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 ()函數(shù)改為如下代碼:
static void Main()
{
if(RunningInstance()==null)
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
else
{
MessageBox.Show("已經運行了一個實例了。");
}
}
方法三:全局原子法,創(chuàng)建程序前,先檢查全局原子表中看是否存在特定原子A(創(chuàng)建時添加的),存在時停止創(chuàng)建,說明該程序已運行了一個實例;不存在則運行程序并想全局原子表中添加特定原子A;退出程序時要記得釋放特定的原子A哦,不然要到關機才會釋放。C#實現(xiàn)如下:
1.申明WinAPI函數(shù)接口
[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()函數(shù)如下:
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"
--------------------------------------*-------*--------*-----------------------------------------------
以上為創(chuàng)建互斥程序的基本通用的思想,個人認為,第一種方法最簡單。
要實現(xiàn)程序的互斥,通常有下面幾種方式,下面用 C# 語言來實現(xiàn):
方法一:
使用線程互斥變量. 通過定義互斥變量來判斷是否已運行實例.
把program.cs文件里的Main()函數(shù)改為如下代碼:
復制代碼 代碼如下:
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);來創(chuàng)建一個互斥體變量run,其中"single_test"為互斥體名,在此方法返回時,如果創(chuàng)建了局部互斥體或指定的命名系統(tǒng)互斥體,則布爾值runone為true;如果指定的命名系統(tǒng)互斥體已存在,則為 false。已命名的互斥體是系統(tǒng)范圍的。
方法二:采用判斷進程的方式,我們在運行程序前,查找進程中是否有同名的進程,同時運行位置也相同程,如是沒有運行該程序,如果有就就不運行.在C#中應用System.Diagnostics名字空間中的Process類來實現(xiàn),主要代碼如下:
1,在program.cs文件中添加函數(shù)如下:
復制代碼 代碼如下:
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 ()函數(shù)改為如下代碼:
復制代碼 代碼如下:
static void Main()
{
if(RunningInstance()==null)
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
else
{
MessageBox.Show("已經運行了一個實例了。");
}
}
方法三:全局原子法,創(chuàng)建程序前,先檢查全局原子表中看是否存在特定原子A(創(chuàng)建時添加的),存在時停止創(chuàng)建,說明該程序已運行了一個實例;不存在則運行程序并想全局原子表中添加特定原子A;退出程序時要記得釋放特定的原子A哦,不然要到關機才會釋放。C#實現(xiàn)如下:
1.申明WinAPI函數(shù)接口
復制代碼 代碼如下:
[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()函數(shù)如下:
復制代碼 代碼如下:
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"
--------------------------------------*-------*--------*-----------------------------------------------
以上為創(chuàng)建互斥程序的基本通用的思想,個人認為,第一種方法最簡單。
相關文章
C#簡單嵌套flash讀取數(shù)據(jù)的實現(xiàn)代碼
這篇文章主要介紹了C#簡單嵌套flash讀取數(shù)據(jù)的實現(xiàn)代碼,有需要的朋友可以參考一下2013-11-11C#設計模式之Mediator中介者模式解決程序員的七夕緣分問題示例
這篇文章主要介紹了C#設計模式之Mediator中介者模式解決程序員的七夕緣分問題,簡單說明了中介者模式的定義并結合七夕緣分問題實例分析了中介者模式的具體使用技巧,需要的朋友可以參考下2017-09-0910分鐘學會Visual Studio將自己創(chuàng)建的類庫打包到NuGet進行引用(net,net core,C#)
這篇文章主要介紹了10分鐘學會Visual Studio將自己創(chuàng)建的類庫打包到NuGet進行引用(net,net core,C#),本文給大家介紹的非常詳細對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09C#實現(xiàn)windows form拷貝內容到剪貼板的方法
這篇文章主要介紹了C#實現(xiàn)windows form拷貝內容到剪貼板的方法,涉及C#操作Clipboard的相關技巧,需要的朋友可以參考下2015-06-06