使用互斥对象让程序只运行一次

原创 2003年08月12日 10:26:00

使用互斥对象让程序只运行一次

“怎么让我的程序在运行时不能重复打开?”经常在论坛上看到有朋友问这方面的问题。本文将比较详细的说明这一问题,并给出一个较为完善的解决方案。

尽管这已经不是一个新问题了,但这里还是简要的说明一下这种技术:这的确是一个相当有用的技术,可能你经常会注意到相当多的程序在运行之后当你再次点击运行时,它只是会回到原来的窗口,而不会运行两个程序。就如同你在运行delphi时,在外部点开另一个工程文件时,delphi只是会简单的将你的当前工程置换而不是运行两个delphi。这样的好处是显而易见的:你不必担心你的程序在某些情况下被别的软件恶意运行多次而吃光内存造成当机。下面我们做进一部的说明:

熟悉win32编程的朋友(特别是多线程编程),相信对互斥对象已经相当熟悉了,它常被用做线程间同步的技术手段。这里我们使用它来防止程序重复运行。我们只是简要的提一下互斥对象,并不做深入研究:互斥对象把第一次建立它的程序作为主程序,这样我们只用检测互斥对象是否已经有主程序就判断程序是否已经运行过,这里需要涉及到一个api函数:WaitForSingleObject该函数的第一个参数为用以检测的互斥对象,第2个参数的表示函数返回结果前的滞留时间,如果改函数返回wait_TimeOut就表明互斥对象已经有了一个主程序。修改了的工程文件代码如下:(注意:以下的代码都出现在工程文件中,而不是单元文件中,并且这里都在最简单的delphi默认建立的工程基础上修改)

var<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

 myMutex:HWND;

begin

  myMutex:=CreateMutex(nil,false,'hkOneCopy');// CreateMutex建立互斥对象,并且给互斥对象起一个唯一的名字。

  if WaitForSingleObject(myMutex,0)<>wait_TimeOut then//程序没有被运行过

  begin

   Application.Initialize;

   Application.CreateForm(TForm1, Form1);

   Application.Run;

  End;

End;

下面的工作是来完善这个程序,我们不仅希望程序可以不被重复运行,而且我们也希望当用户再次点击程序可执行文件时,已经运行的程序能够做出一些响应。在这里我们希望它能够变为最上层的活动窗口以提醒用户程序已经被运行。为了达到这个目的,我们必须先获得已经运行程序的窗口句柄,以便使用SetForeGroundWindow(handle)来使程序窗口最前并激活。为了得到这个句柄,我们必须使用windows枚举函数EnumWindows来遍历windows的窗口列表,该函数可以使用一个回调函数作为参数,并用这个回调函数来对每一个系统中的窗口进行调用直到最后一个窗口或回调函数返回false为止,这个回调函数规定有两个参数(handle,Cardinal,只用注意第一个handle参数它表示由枚举函数当前遍历到的窗口句柄)。我们只要编写这个函数并在其中不断的比较当前遍历到的窗口类名和我们的程序的主窗口类名,以及比较窗口可执行文件的名称和我们程序的名称直到找到相同的为止,将这时的窗口句柄保存下来就可以了,下面的代码加上了适当的注释:

function EnumWndProc(hwnd:Thandle;param:Cardinal):bool;stdcall;

//由于用于api回调函数,请使用windows传统的参数传递方式stdcall

var

 ClassName,WinMoudleName:string;

 WinInstance:THandle;

begin

 result:=true;

 SetLength(ClassName,100);

 GetClassName(hwnd,pchar(ClassName),length(ClassName));//获得当前遍历窗口的类名

 ClassName:=pchar(ClassName);//在字符串后加结束符,确定字符串结束

 if ClassName=TForm1.ClassName then//比较

 begin

  WinInstance:=GetWindowLong(hwnd,GWL_HINSTANCE);//获得当前遍历窗口的实例

  setlength(WinMoudleName,100);

  GetModuleFileName(WinInstance,pchar(WinMoudleName),length(WinMoudleName));

  //获得当前遍历窗口的程序文件名

  WinMoudleName:=pchar(WinMoudleName);

  if WinMoudleName=MoudleName then//MoudleName为工程全局变量,自身程序的文件名

  begin

   FindHid:=hwnd;//FindHid为工程全局变量保存找到的句炳

   result:=false;//找到以后就结束遍历

  end;

 end;

end;

下面是全部的工程文件:

var

 hMutex,FindHid:HWND;

 MoudleName:string;

begin

  hMutex:=CreateMutex(nil,false,'hkOneCopy');

  if WaitForSingleObject(hMutex,0)<>wait_TimeOut then

  begin

   ……//略去的代码在前文

  end

  else

  begin

   SetLength(MoudleName,100);

   GetModuleFileName(HInstance,pchar(MoudleName),length(MoudleName));

   //获得自己程序文件名

   MoudleName:=pchar(MoudleName);

   EnumWindows(@EnumWndProc,0);//调用枚举函数

   if FindHid<>0 then

    SetForegroundWindow(FindHid);

  end;

end.

为了使我们的程序更完美,让它能在重复运行的时候展现更多的特性(如delphi中的置换工程文件为当前打开的工程),你还可以向找到的窗口句柄发送用户消息,再在窗口的消息处理函数中做相应的处理,你一定可以让我们的程序更眩!

参考文献:

   delphi开发者指南》

只允许运行一个实例的程序(互斥程序)

有的时候我们需要编写出的应用程序在同一时间只允许运行一个实例,运行第二个实例时提示用户“该程序已经运行”。这是很老土的表示方法,但很灵验。我找到了三种限制程序实例被多次运行的方法,它们有各自的优点,一...
  • dadahacker
  • dadahacker
  • 2009年08月29日 12:03
  • 1213

MFC基于对话框程序只让程序启动一次

这个功能一直都有在用,以前没有记录下来,所以今天补上。 首先,用VS或者VC下的GUIDGEN生成一个唯一的标示供程序使用,然后在APP类头文件定义一个HANDLE变量,在APP实现类加入以下代码即可...
  • sky1718
  • sky1718
  • 2016年03月11日 17:33
  • 323

VC中如何让一个应用程序只启动一次

方法一:使用#pragma data_seg 步骤一:在应用程序的入口文件前面加上如下语句 #pragma data_seg("flag_data") int g_Count = 0; //共...
  • youyongyoumou
  • youyongyoumou
  • 2015年04月09日 10:42
  • 2003

怎样让VB6程序只能运行一次

方法一:利用PrevInstance属性 If App.PrevInstance Then Call MsgBox("对不起本程序已在运行中, 不得重复加载!!", vbCritical) End E...
  • myinc
  • myinc
  • 2013年12月05日 17:02
  • 1305

如何让程序只运行一次实例

using System;using System.Runtime.InteropServices;using System.Windows.Forms;using System.Diagnostic...
  • abaowu
  • abaowu
  • 2004年11月22日 12:04
  • 1002

使用互斥对象让程序只运行一次(delphi)

使用互斥对象让程序只运行一次“怎么让我的程序在运行时不能重复打开?”经常在论坛上看到有朋友问这方面的问题。本文将比较详细的说明这一问题,并给出一个较为完善的解决方案。尽管这已经不是一个新问题了,但这里...
  • formiss
  • formiss
  • 2006年12月26日 09:50
  • 1701

如何使一个程序只运行一个

方法一:If App.PrevInstance = True Then MsgBox "程序已经在运行中,不打开两个!", vbCritical + vbOKOnly, "友情提示"  方法二:设计时...
  • surro
  • surro
  • 2007年08月20日 15:13
  • 322

android只让某个activity运行一次-SharedPreferences

使用SharedPreference来完成。SharedPreferences是Android平台上一个轻量级的存储类,用来保存应用的一些常用配置 | 只让进入一次引导页         ...
  • u012246458
  • u012246458
  • 2015年06月25日 14:40
  • 1280

创建互斥对象,让程序只运行一个。

int _tmain(int argc, _TCHAR* argv[]) { // 创建互斥量 不允许多开 HANDLE m_hMutex = CreateMutex(NULL, FAL...
  • fanpeii
  • fanpeii
  • 2012年10月25日 09:56
  • 446

让程序只运行一次

 使应用程序只能运行一个实例此方法对于对话框程序或者单文档程序等都适合    Windows是多进程操作系统,框架生成的应用程序可以多次运行,形成多个运行实例。但在有些情况下为保证应用程序的安全运行,...
  • lcrlcr
  • lcrlcr
  • 2008年11月05日 16:31
  • 254
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:使用互斥对象让程序只运行一次
举报原因:
原因补充:

(最多只允许输入30个字)