博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
delphi 防止程序重复执行
阅读量:3588 次
发布时间:2019-05-20

本文共 6315 字,大约阅读时间需要 21 分钟。

实现单实例运行的关键是判断前一实例是否存在,Win3.x中运行的程序能获知前

一实例的句柄,从而可以方便地进行判断,但 Windows 95 是抢先式多任务系统,其

程序的前一实例句柄恒为零,所以只有另寻其他办法。目前最有效的办法是通过查看

是否有相同窗口类名的例程存在来进行判断。下面介绍在Delphi中实现的方法。


1、对主窗口程序的改动:


在主窗口(即程序创建的第一个窗口)中interface节加入

constCM_RESTORE = WM_USER + $1000; {自定义的“恢复”消息}MYAPPNAME = "My Delphi Program";并在Form的定义的public节中加入procedure CreateParams(var Params: TCreateParams); override;Procedure RestoreRequest(var message: TMessage); message CM_RESTORE;在implementation节中加入{指定窗口名称}procedure TForm1.CreateParams(var Params: TCreateParams);begininherited CreateParams(Params);Params.WinClassName := MYAPPNAME;end;{处理“恢复”消息}procedure TForm1.RestoreRequest(var message: TMessage);beginif IsIconic(Application.Handle) = TRUE thenApplication.RestoreelseApplication.BringToFront;end;

经过以上修改,程序的主窗口的类名已经被指定了,这是进行判断的基础。一般在程

序刚开始运行的时候进行判断,所以还要对DPR文件进行修改。


2、对DPR文件的改动


在 uses 节中添加 windows、messages这两个单元加入下列语句,注意两个文件中常

量CM_RESTORE和MYAPPNAME的定义必须一致

const

CM_RESTORE = WM_USER + $1000; {自定义的“恢复”消息}

MYAPPNAME = "My Delphi Program";

var

RvHandle : hWnd;


将下列语句插到程序最前部(在Application.Initialize之前)


RvHandle := FindWindow(MYAPPNAME, NIL);

if RvHandle > 0 then

begin

PostMessage(RvHandle, CM_RESTORE, 0, 0);

Exit;

end;

这段程序的意思是如果找到一个类名相同的窗口,则向该窗口发送一个消息,并退

出,而本例中原窗口收到该消息后会自动激活或从图标还原,从而达到了避免二次运

行且能自动调出前一例程的目的。


----------------------------------------------------------------


方法三: 工程文件中:

{$R *.res}


const mypro='tmainapp';//主窗体类

var handle:integer;

begin

handle:=findwindow(mypro,nil);

if handle<>0 then

begin

messagebox(0,'程序正在运行,请退出!','警告!',0);

// halt;

end;

Application.CreateForm(Tmainapp, mainapp);

------------------------------------------------------------------------------


方法四:


procedure Tloginform.FormCreate(Sender: TObject);

var

errno:integer;

hmutex:hwnd;

begin

hmutex:=createmutex(nil,false,pchar(application.Title));

errno:=getlasterror;

if errno=error_already_exists then

begin

application.MessageBox(' 您已经打开了该软件'+#13#13+' 请不要再尝试'+#13#13+'您只能运行一个程序实例','不要试图打开多个',mb_ok);

application.Terminate;

end;

end; 

------------------------------------------------------------

在《Delphi 5 开发人员指南》中第13章中有一篇"防止同时出现多个应用程序实例",

代码中给出了一个MultInst.pas单元,工程引用此单元就能防止同时出现多个实例,

但实际应用中发现,如果应用程序并没有最小化,第二个实例不能把第一个实例提到最前.

下面是我改写的MultInst.pas单元,能解决这个小问题.

//==============================================================================

// Unit Name: MultInst

// Author : ysai

// Date : 2003-05-20

// Purpose : 解决应用程序多实例问题

// History :

//==============================================================================


//==============================================================================

// 工作流程

// 程序运行先取代原有向所有消息处理过程,然后广播一个消息.

// 如果有其它实例运行,收到广播消息会回发消息给发送程序,并传回它自己的句柄

// 发送程序接收到此消息,激活收到消息的程序,然后关闭自己

//==============================================================================

unit MultInst;


interface


uses

Windows ,Messages, SysUtils, Classes, Forms;


implementation


const

STR_UNIQUE = '{2BE6D96E-827F-4BF9-B33E-8740412CDE96}';

MI_ACTIVEAPP = 1; //激活应用程序

MI_GETHANDLE = 2; //取得句柄


var

iMessageID : Integer;

OldWProc : TFNWndProc;

MutHandle : THandle;

BSMRecipients : DWORD;


function NewWndProc(Handle: HWND; Msg: Integer; wParam, lParam: Longint):

Longint; stdcall;

begin

Result := 0;

if Msg = iMessageID then

begin

case wParam of

MI_ACTIVEAPP: //激活应用程序

if lParam<>0 then

begin

//收到消息的激活前一个实例

//为什么要在另一个程序中激活?

//因为在同一个进程中SetForegroundWindow并不能把窗体提到最前

if IsIconic(lParam) then

OpenIcon(lParam)

else

SetForegroundWindow(lParam);

//终止本实例

Application.Terminate;

end;

MI_GETHANDLE: //取得程序句柄

begin

PostMessage(HWND(lParam), iMessageID, MI_ACTIVEAPP,

Application.Handle);

end;

end;

end

else

Result := CallWindowProc(OldWProc, Handle, Msg, wParam, lParam);

end;


procedure InitInstance;

begin

//取代应用程序的消息处理

OldWProc := TFNWndProc(SetWindowLong(Application.Handle, GWL_WNDPROC,

Longint(@NewWndProc)));


//打开互斥对象

MutHandle := OpenMutex(MUTEX_ALL_ACCESS, False, STR_UNIQUE);

if MutHandle = 0 then

begin

//建立互斥对象

MutHandle := CreateMutex(nil, False, STR_UNIQUE);

end

else begin

Application.ShowMainForm := False;

//已经有程序实例,广播消息取得实例句柄

BSMRecipients := BSM_APPLICATIONS;

BroadCastSystemMessage(BSF_IGNORECURRENTTASK or BSF_POSTMESSAGE,

@BSMRecipients, iMessageID, MI_GETHANDLE,Application.Handle);

end;

end;


initialization

//注册消息

iMessageID := RegisterWindowMessage(STR_UNIQUE);

InitInstance;


finalization

//还原消息处理过程

if OldWProc <> Nil then

SetWindowLong(Application.Handle, GWL_WNDPROC, LongInt(OldWProc));


//关闭互斥对象

if MutHandle <> 0 then CloseHandle(MutHandle);


end.


------------------------------------------------


Windows 下一个典型的特征就是多任务,我们可以同时打开多个窗口进行操作,也可以同时运行程序的多个实例,比如可以打开许多个资源管理器进行文件的移动复制操作。但有时出于某种考虑(比如安全性),我们要做出一些限制,让程序只能够运行一个实例。在Delphi编程中,笔者总结出了以下几种方法:

  一、 查找窗口法

  这是最为简单的一种方法。在程序运行前用FindWindow函数查找具有相同窗口类名和标题的窗口,如果找到了,就说明已经存在一个实例。在项目源文件的初始化部分添加以下代码:

  Program OneApp

  Uses

  Forms,Windows;(这里介绍的几种方法均需在项目源文件中添加Windows单元,以后不再重复了)

  Var Hwnd:Thandle;

  Begin

   Hwnd:=FindWindow(‘TForm1’,‘SingleApp’);

   If Hwnd=0 then

   Begin

   Application.Initialize;

   Application.CreateForm(Tform1, Form1);

   Application.Run;

   End;

  End;

  FindWindow()函数带两个参数,FindWindow的第一个参数是类名,第二个参数是窗口标题,其中的一个参数可以忽略,但笔者强烈建议将两个参数都用上,免得凑巧别的程序也在使用相同的类名,就得不到正确的结果了。另外,如果是在Delphi IDE窗口中运行该程序,将一次都不能运行,因为已经存在相同类名和标题的窗口:设计时的窗体。

 
 二、使用互斥对象
  如果觉得查找窗口的方法效率不太高的话,可以使用创建互斥对象的方法。尽管互斥对象通常用于同步连接,但用在这个地方也是非常方便的。仅用了4句代码就轻松搞定。
  VAR Mutex:THandle;
  begin
   Mutex:=CreateMutex(NIL,True,‘SingleApp’);
   IF GetLastError<>ERROR_ALREADY_EXISTS THEN//如果不存在另一实例
   BEGIN
   Application.CreateHandle;
   Application.CreateForm (TExpNoteForm, ExpNoteForm);
   Application.Run;
   END;
   ReleaseMutex(Mutex);
  end.
  三、全局原子法

  我们也可以利用向系统添加全局原子的方法,来防止多个程序实例的运行。全局原子由Windows 系统负责维持,它能保证其中的每个原子都是唯一的,管理其引用计数,并且当该全局原子的引用计数为0时,从内存中清除。我们用GlobalAddAtom 函数向全局原子添加一个255个字节以内的字符串,用GlobalFindAtom来检查是否已经存在该全局原子,最后在程序结束时用GlobalDeleteAtom函数删除添加的全局原子。示例如下:

  Uses Windows

  const iAtom=‘SingleApp’;

  begin

   if GlobalFindAtom(iAtom)=0 then

   begin

   GlobalAddAtom(iAtom);

   Application.Initialize;

   Application.CreateForm(TForm1,Form1);

   Application.Run;

   GlobalDeleteAtom(GlobalFindAtom(iAtom));

   end

   else

   MessageBox(0,‘You can not run a second copy of this App’,‘’,mb_OK);

  end.

  利用全局原子的引用计数规则,我们还可以判断当前共运行了该程序的多少个实例:

  var i:Integer;

  begin

   I:=0;

  while GlobalFindAtom(iAtom)<>0 do

   begin

   GlobalDeleteAtom(GlobalFindAtom(iAtom));

   i:=i+1;

   end;

   ShowMessage(IntToStr(I));

  end;

转载地址:http://tmvwn.baihongyu.com/

你可能感兴趣的文章
Struts2原理
查看>>
activemq总结
查看>>
jdk环境变量配置
查看>>
mybatis原理
查看>>
spring原理
查看>>
神州炫龙蓝天刷bios教程(百分百成功/不成功下面留言)
查看>>
java里VO是什么?
查看>>
AOP
查看>>
JDK和JRE的区别
查看>>
zookeper正式集群搭建(非伪集群)
查看>>
linux定时备份mysql(可用)
查看>>
mysql环境变量
查看>>
linux使用链接下载文件
查看>>
maven配置阿里云仓库
查看>>
配置maven本地仓库路径
查看>>
idea生成mybatis实体的方法
查看>>
idea逆向工程mybatis
查看>>
oracle纯url连接字符串
查看>>
oracle自动提交事务以及手动
查看>>
CPU温度原理
查看>>