zoukankan      html  css  js  c++  java
  • [转]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
      ....
  • 相关阅读:
    0xc000000f: Error attempting to read the boot configuration data
    结合使用 Oracle Database 11g 和 Python
    精通 Oracle+Python,第 9 部分:Jython 和 IronPython — 在 Python 中使用 JDBC 和 ODP.NET
    精通 Oracle+Python,第 8 部分:适合 Oracle DBA 使用的 Python
    精通 Oracle+Python,第 6 部分:Python 支持 XML
    精通 Oracle+Python,第 7 部分:面向服务的 Python 架构
    精通 Oracle+Python,第 5 部分:存储过程、Python 编程
    精通 Oracle+Python,第 4 部分:事务和大型对象
    精通 Oracle+Python,第 3 部分:数据解析
    精通 Oracle+Python,第 2 部分:处理时间和日期
  • 原文地址:https://www.cnblogs.com/go-jzg/p/4196800.html
Copyright © 2011-2022 走看看