[转]Delphi中,让程序只运行一次的方法

program onlyRunOne;

uses
  Forms,Windows,SysUtils, Dialogs,

  Unit1 in ‘Unit1.pas‘ {Form1};

{$R *.res}
var
myMutex:HWND;

begin
myMutex:=CreateMutex(nil,false,‘11111‘);     //名称只能全系统唯一。
if WaitForSingleObject(myMutex,0)<>wait_TimeOut then

begin
  Application.Initialize;
  Application.CreateForm(TForm1, Form1);
  Application.Run;
  end else begin
            Application.MessageBox(‘已经有一个事例运行了,本事例不能运行‘,‘提示‘,mb_ok+mb_IconInformation);
          Application.Terminate;
  end;

end.

hMutex:=CreateMutex(nil,true,‘test1‘);
      if   GetLastError=ERROR_ALREADY_EXISTS   then
      begin
          Application.MessageBox(‘已经有一个事例运行了,本事例不能运行‘,‘提示‘,mb_ok+mb_IconInformation);
          Application.Terminate;
          Abort;
      end;
//其中 test1 是随便 起的字符,但如果别的程序也用这个名字 ,那2个程序也只能找开一个哦。

网上找了很多方法,很多有漏洞

自己总结实验了出来 下面是正确的

program Project1; 

uses
  Forms,Windows,SysUtils,Dialogs,
  Unit1 in ‘Unit1.pas‘ {Form1},
  Unit2 in ‘Unit2.pas‘ {Form2},
  Unit3 in ‘Unit3.pas‘ {Form3}; 

{$R *.res} 

var
myMutex:HWND;
begin
myMutex:=CreateMutex(nil,false,‘hkOneCopy‘);
if WaitForSingleObject(myMutex,0)<>wait_TimeOut then
begin 

Application.Initialize;
Application.CreateForm(TForm1, Form1);
Application.CreateForm(TForm2, Form2);
Application.CreateForm(TForm3, Form3);
Application.Run; 

end
else
begin
showmessage(‘你已经运行了程序在屏幕右下角处‘);
end; 

end.

  多实例指同时有同一个应用程序的多个副本在运行。同一个应用程序的多个副本可以相互独立地同时运行,是Win32操作系统提供的一个功能。但有时,我们可能希望用户启动应用程序后就不再启动它的别的副本。比如某种设备资源的控制程序,像调制解调器和并行端口。这种情况下,用程序代码防止同时出现多个程序的副本在运行是非常必要的。
  在16位的Windows中,要防止出现多个实例是很简单的,因为系统变量hPrevInst可以被用来判断是否有其他的实例存在。当hPrevInst变量不为0时,表示已经有别的应用程序实例在运行。
  然而,在Win32系统中每个进程之间有R32绝缘层来彼此隔绝。因此,在Win32系统中变量hPrevInst的值总为0。另一种既适合Win32系统又适合于16位的Windows的技术,是调用FindWindow()API函数去搜索一个已激活的程序窗口。
  Windows API 提供了函数FindWindow,可以是应用程序在启动时检查自己是否已经存在。 该函数在Delphi中的语法为:
  function FindWindow(lpClassName: PChar, lpWindowName: PChar): HWND;
  其中,参数lpCalssName 是要查找的窗口的类的名称,参数lpWindowName是要查找的窗口的标题(Caption)。 如果找到了相应的窗口实例,将返回一个非0 的该窗口句柄的整型值,否则返回0 。因此,只要判断应用程序的主窗口(或者伴随着应用程序存在而存在的窗口)是否存在就可以判断是否已经有实例存在了。
  例如: H := FindWindow(‘TForm1‘, nil);
  if H = 0 then begin
    ShowMessage(‘没有发现相同的应用程序实例。‘);
   //加入加载应用程序的语句
   //
  end else begin
   ShowMessage(‘应用程序已经加载。‘);
   SetActiveWindow(H);
  end;其中,参数lpWindowName的位置以Delphi保留字nil 代替,是因为窗口的标题可能在应用程序中是变化的。Windows API 函数SetActiveWindow 用于指定活动窗口。

  但是,这种方法有两个缺陷:一是它只能基于窗口类名或标题来搜索窗口,但是在整个系统中窗口很可能会重复。所以,这样做是不可靠的。而利用窗口的标题的方法也有问题,因为窗口的标题有可能发生变化(以Delphi和Word为例,每次打开不同文件,它们的标题都会变化),所以这种方法不可取。另一个缺陷是它每次搜索都要遍历所有窗口,这样执行进来非常慢。

  因此,在Win32系统中最好的解决方案是利用那些不依赖于进程的API对象,并且它们的使用也很简单,互斥对象就可以解决这个问题。当一个应用程序首次运行时,我们就使一个互斥对象被API函数CreateMutex()创建。这个函数的参数lpName是一个唯一标识互斥对象的字符串。当应用程序的实例要运行前,它首先要用OpenMutex()来打开互斥对象,如果已经有一个CreateMutex()创建的互斥对象则返回非零值。另外,当试图运行另一个程序实例时,使第一个实例被激活。
  对于这个问题,最好的解决方法是在首次运行时,利用RegisterWindowMessage()函数注册一个消息,并在应用程序中创建唯一的消息标识符。然后,利用第一个实例对这个消息的响应使它被第二个实例激活。

    这种方法阻止新实例的产生,但不能提前,不过较简便。

在Project的Program文件中program Live;   

uses
  Windows,
  Forms,
  ShellApi,
  SysUtils,
  ..;   

{$R *.TLB}  

{$R *.res}
var
  HMutex:Hwnd;
  Ret:Integer;
begin  

  Application.Initialize;
  aTitle := ‘LiveAuction‘;
  Application.Title := ‘LiveAuction‘;   

  HMutex:=CreateMutex(nil,False,Pchar(aTitle));  //建立互斥对象,名字为aTitle--‘LiveAuction‘
  Ret:=GetLastError;
  If Ret<>ERROR_ALREADY_EXISTS Then
  begin
     //做我们正常该做的事情
  end else
    ReleaseMutex(hMutex);  //防止创建多个程序实例   

  Application.Run;
end.
检查某个exe文件是否正在运行
function exe_is_running(const exeName:String) : Boolean;  //exeName:不要扩展名的Exe主文件名
var
  hCurrentWindow:HWnd;
  szText:array[0..254] of char;
begin
  Result := False;
  hCurrentWindow:=Getwindow(Application.Handle,GW_HWNDFIRST);
  while hCurrentWindow <> 0 do
  begin
    if Getwindowtext(hCurrentWindow,@sztext,255)>0 then
    begin
       if LowerCase(pchar(@sztext))=LowerCase(exeName) then
       begin
         Result := true;
         Exit;
       end;
    end;
    hCurrentWindow:=Getwindow(hCurrentwindow,GW_HWndNext);
  end;
end;
用法:
如我们要判断‘Live.exe‘程序是否正在运行/是否已经启动
if exe_is_running(Live) then
  ....
else
  ....
时间: 2024-08-24 15:08:53

[转]Delphi中,让程序只运行一次的方法的相关文章

[VC]在VC++中实现让程序只运行一个实例的方法且实现该实例

方法一: 有时候在开发应用程序时,希望控制程序运行唯一的实例.例如,最常用的mp3播放软 件Winamp,由于它需要独占计算机中的音频设备,因此该程序只允许自身运行唯一的一个例程.在Visual C++的开发实践中,对于16位的Windows系统,应用程序的hPrevInstance句柄保存了应用程序上一个运行的实例,可以用该值来检查是否 有实例运行:然而在32位Windows系统下,这个值总是NULL,所以无法利用该值来实现程序运行唯一实例.本实例给出了解决这个问题的简单办法,只 要将程序中稍

关于WPF程序只运行一个实例的方法

找到的方法有两种: 1)http://www.cnblogs.com/liuyazhou/archive/2009/11/02/1594364.html 2)http://codereview.stackexchange.com/questions/20871/single-instance-wpf-application

vc++高级班之窗口篇[4]---让程序只运行一个实例

大家都看过或者使用过类似只运行一个实例的程序,比如:QQ游戏.部分浏览器 等等! 让一个程序只运行一个实例的方法有多种,但是原理都类似,也就是在程序创建后,有窗口的程序在窗口创建前, 检查系统中是否已经设置了某些特定标志,是否创建了一些全局唯一的东西,或者让程序的多个实例都能看到的东西, 如果有则说明已经有一个实例在运行了,则当前程序通知用户如何如何,然后程序退出,当然方法有很多种,各有各的优缺点! ①.创建互斥体 Mutex 法: 但是单纯的使用互斥体的话不能取得已经创建的实例窗口局柄,因此无

WINCE程序只运行一个,防止多重运行。

在 winform中,程序只运行一个,防止多重运行,很容易.如用FindWindow.Mutex和C#直接Process遍历,都可以实现. 但是,到了WINCE 系统中,要么方法不存在,即使引入CreateMutex,也是白搭.根本无效. 从网上找到一篇通过系统快照方式的方案.不过,大侠们写的都不完整,我特地贡献一个可以直接使用的类: using System.IO; using System.Runtime.InteropServices; //可能还缺其他using,,,,没仔细看. pub

DELPHI中枚举类型数据的介绍和使用方法

在看delphi程序的时候看到aa=(a,b,c,d);这样的东西,还以为是数组,同事说是函数,呵呵,当然这两个都不屑一击,原来这样式子是在声明并付值一个枚举类型的数据.下边写下来DELPHI中枚举类型数据的介绍和使用方法.见笑..------------------------------------------------------枚举类型 Pascal程序不仅用于数值处理,还更广泛地用于处理非数值的数据.例如,性别.月份.星期几.颜色.单位名.学历.职业等.1.枚举类型的定义格式: ty

Delphi中Twebbrowser的document 对象的属性、方法、事件一览(转)

Delphi中Twebbrowser的document 对象的属性.方法.事件一览(转) 2012-12-07 10:19:39|  分类: Delphi 零碎 |  标签:webbrowser  属性  方法  事件  delphi   |举报 |字号大中小 订阅 {ihtmldocument2 方法:}write //写入writeln //写入并换行open //打开一个流,以收集 document.write 或 document.writeln 的输出close //关闭并输出用 do

JAVA中的反射只获取属性的get方法

JAVA中的反射只获取属性的get方法 在开发过程中,我们经常需要获取类中某些属性的get方法,此时我们需要使用到反射,例如,我们在获得一个对象后,需要知道该对象的哪些属性有值,哪些没有值,方便我们后面的处理. 譬如在我们拼SQL语句时,就需要知道哪些字段为空或为NULL,此时我们在拼语句的时候需要剔除掉,若是我们采用一般的判断的办法,则会很复杂(需要处理好SQL中的AND关键字的有无 ),当然,我们也有另外的解决办法(例如将非空的键和值存入map中,再将map存入list集合中,然后循环集合做

让 Delphi 程序只运行一个副本

program mutex; uses  Windows,  Forms,  Unit1 in 'Unit1.pas' {Form1}; {$R *.res} const  MutexStr = '4C3201D5-6A5A-4B8B-A2F0-B103985705F4'; // GUID var  hAppMutex: THandle; begin  ReportMemoryLeaksOnShutdown := DebugHook = 1;  Application.Initialize; 

C#让应用程序只运行一个实例的几种方法

一 判断是否有相同的实例已经运行 1 根据“Mutex”判断是否有相同的实例在运行 /// <returns>已有实例运行返回true,否则为false</returns>public bool IsRunningProcessByMutex(){     bool createNew;     using (System.Threading.Mutex mutex = new System.Threading.Mutex(true, Application.ProductName