如果在你的軟件中留下了你的網頁地址和郵件地址,你肯定希望人們點擊它就會啟動浏覽器或者電子郵件軟件。這其實就是如何啟動外部軟件的問題,很簡單,不是嗎?不過,如果我問你,如何啟動外部程序並等待它結束,你還能告訴我嗎?
其實,這是一個“古老”的話題,在WIN95時代就被討論過了。不過,既然這麼多人不知道,我感覺還是有必要再討論一下。
一、為什麼要啟動外部程序
也許,你想讓你的程序完成全部的功能。不過,無論從物力還是人力上,你都應養成資源共享的習慣。更好的考慮是,充分利用已有的程序,而讓你的程序專注於某一方面的功能。比如說,浏覽器負責打開網頁,讓人們浏覽,當遇到下載的任務時,可以交給更專業的下載軟件去做。你也可能在你的程序裡留下了你的主頁和郵箱地址,你希望有人點擊它們時就分別啟動浏覽器和電子郵件。在某些情況下,你需要外部程序處理後,再進行下一步的工作,這時就會遇到啟動外部程序並等待它結束的問題。
二、預備知識
啟動外部程序我們可以使用函數Winexec、ShellExecute和ShellExecuteEx。我推薦大家使用函數ShellExecute,因為它既靈活,又簡單。看看下面的例子,用法就清楚了:
*: 啟動一個程序
ShellExecute(Handle,'open',PChar('c: estapp.exe'),
nil,nil,SW_SHOW);
* 啟動記事本 (因為記事本在系統路徑下,所以不必寫完整的路徑名了):
ShellExecute(Handle, 'open', PChar('notepad'),
nil, nil, SW_SHOW);
* 啟動記事本並加載一個純文本文件:
ShellExecute(Handle, 'open', PChar('notepad'),
PChar('c: est eadme.txt', nil, SW_SHOW);
* 使用記事本打開一個純文本文件 (請確定*.txt文件被關聯到記事本):
ShellExecute(Handle, 'open', PChar('c: est eadme.txt'),
nil, nil, SW_SHOW);
* 使用默認浏覽器打開網址:
ShellExecute(Handle, 'open', PChar('http://www.festra.com/'),
nil, nil, SW_SHOW);
* 打印一個文件:
ShellExecute(Handle, 'print', PChar('c: est eadme.txt'),
nil, nil, SW_SHOW);
* 用Windows Explorer打開一個文件夾:
ShellExecute(Handle, 'explore', PChar('c:Windows)',
nil, nil, SW_SHOW);
* 運行一個DOS命令並立即返回:
ShellExecute(Handle, 'open', PChar('command.com'),
PChar('/c copy file1.txt file2.txt'), nil, SW_SHOW);
* 運行一個DOS命令並保持DOS窗口打開 ("stay in DOS"):
ShellExecute(Handle, 'open', PChar('command.com'),
PChar('/k dir'), nil, SW_SHOW);
啟動一個外部程序並不難吧?不過,我們如何知道它是否運行結束了呢?我們的程序又怎樣等待它結束呢?
三、啟動外部程序並等待它結束的函數
我們可以通過進程句柄(process handle)來查看進程(程序)是否結束。為了得到進程句柄,有兩個Win32 API函數可以利用:ShellExecuteEx 或者CreateProces。解決這個問題最簡單的方法是,使用ShellExecuteEx啟動一個外部程序,然後使用WaitForSingleObject管理這個外部程序的進程句柄。我們可以這樣定義一個函數:
……
{ ExecAppWait:功能:運行外部程序並等待它結束。。
運行外部程序APPNAME,參數PARAMS;
Returns:如果外部程序出錯返回 FASLE
}
function ExecAppWait(AppName, Params: string): Boolean ;
……
function ExecAppWait(AppName, Params: string): Boolean;
var
// Structure containing and receiving info about application to start
ShellExInfo: TShellExecuteInfo;
begin
FillChar(ShellExInfo, SizeOf(ShellExInfo), 0);
with ShellExInfo do begin
cbSize := SizeOf(ShellExInfo);
fMask := see_Mask_NoCloseProcess;
Wnd := Application.Handle;
lpFile := PChar(AppName);
lpParameters := PChar(Params);
nShow := sw_ShowNormal;
end;
Result := ShellExecuteEx(@ShellExInfo);
if Result then
while WaitForSingleObject(ShellExInfo.HProcess, 100) = WAIT_TIMEOUT do
begin
Application.ProcessMessages;
if Application.Terminated then Break;
end;
end;
……
不難理解吧?
建立一個Unit ExecWait,把上面的代碼輸進去。
四、例子
如圖建立Form,源程序如下:
unit DemoUnit;
interface
uses
Windows, Messages, SysUtils, Classes, Graphics, Controls,
Forms, Dialogs, StdCtrls, SHELLAPI;
type
TForm1 = class(TForm)
Edit1: TEdit;
Edit2: TEdit;
Label1: TLabel;
Label2: TLabel;
BtnExec: TButton;
CheckBoxWait: TCheckBox;
Label3: TLabel;
Label4: TLabel;
Edit3: TEdit;
Edit4: TEdit;
Label5: TLabel;
procedure BtnExecClick(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form1: TForm1;
implementation
uses
execwait;
{$R *.DFM}
procedure TForm1.BtnExecClick(Sender: TObject);
var
Success: Boolean;
InstanceID: THandle;
begin
{ 最小化窗口,提醒發生的變化 }
Application.Minimize;
Success := False;
try
if CheckBoxWait.Checked then
Success := ExecAppWait(Edit1.Text, Edit2.Text)
else begin
InstanceID := ShellExecute(Handle, 'open', PChar(Edit1.Text),
PChar(Edit2.Text), nil, SW_SHOW);
Success := InstanceID >= 32; // 小於32可就錯了
end;
finally
// 可別忘了恢復我們的程序的窗口!
Application.Restore;
if not Success then
ShowMessage('Application 1 failed: ' + Edit1.Text + ' ' + Edit2.Text);
end;
try
if CheckBoxWait.Checked then
Success := ExecAppWait(Edit3.Text, Edit4.Text)
else begin
InstanceID := ShellExecute(Handle, 'open', PChar(Edit3.Text),
PChar(Edit4.Text), nil, SW_SHOW);
Success := InstanceID >= 32; //小於32可就錯了
end;
finally
//恢復我們的程序的窗口
Application.Restore;
if not Success then
ShowMessage('Application 2 failed: ' + Edit3.Text + ' ' + Edit4.Text);
end;
end;
end.
OK,沒有問題吧?你趕快試試吧,把它應用到你的程序裡。#