Windows 下一个典型的特征就是多任务,我们可以同时打开多个窗口进行操作,也可以同时运行程序的多个实例,比如可以打开许多个资源管理器进行文件的移动复制操作。但有时出于某种考虑(比如安全性),我们要做出一些限制,让程序只能够运行一个实例。在Delphi编程中,笔者总结出了以下几种方法: 一、 查找窗口法 这是最为简单的一种方法。在程序运行前用FindWindow函数查找具有相同窗口类名和标题的窗口,如果找到了,就说明已经存在一个实例。在项目源文件的初始化部分添加以下代码:
- Program OneApp
- Uses
- Forms,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()函数带两个参数,其中的一个参数可以忽略,但笔者强烈建议将两个参数都用上,免得凑巧别的程序也在使用相同的类名,就得不到正确的结果了。另外,如果是在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://blog.csdn.net/chaijunkun/article/details/5574332
|
请发表评论