背景说明:
前段时间开发一个数据转换的系统,业务逻辑中说明数据需要压缩成.tar.gz格式。
我在Windows系统下采用,先生成批处理文件,然后调用WinExec执行批处理文件,休眠等待一段时间,完成数据的自动压缩。
后来发现,待压缩文件的大小不确定,单纯的执行WinExec时Sleep固定时间,可能导致压缩失败、文件不全或损坏。
优化方案:
取代WinExe用CreateProcess用来启动进程, 执行批处理文件, 同时系统会自动填写TProcessInformation这个结构。
此时程序会自动阻塞到该批处理中,等待批处理句柄的进程结束或超时。这样就能解决压缩损坏问题。
给个实例Demo:
D7代码如下:
unit uMain; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, RzButton, StdCtrls; type TFrmMain = class(TForm) mmMsg: TMemo; btnExecute: TRzBitBtn; btnClear: TRzBitBtn; procedure MsgDsp(v_Str: string); procedure btnExecuteClick(Sender: TObject); procedure btnClearClick(Sender: TObject); private { Private declarations } public { Public declarations } end; var FrmMain: TFrmMain; implementation {$R *.dfm} procedure TFrmMain.MsgDsp(v_Str: string); begin mmMsg.Lines.Add('[ admin ] - [' + v_Str + '] - [' + FormatDateTime('YYYY-MM-DD hh:mm:ss zzz', Now()) + ']'); end; procedure TFrmMain.btnExecuteClick(Sender: TObject); var sInfo: TStartupInfo; pInfo: TProcessInformation; cmdLine: string; exitCode: Cardinal; begin MsgDsp('初始化参数'); cmdLine := 'C:\Program Files\7-Zip\7zFM.exe'; FillChar(sInfo, sizeof(sInfo), #0); sInfo.cb := SizeOf(sInfo); sInfo.dwFlags := STARTF_USESHOWWINDOW; sInfo.wShowWindow := SW_NORMAL; MsgDsp('参数初始化完成,启动WinExec调试'); //CreateProcess用来启动进程, 进程启动后, 会填写TProcessInformation这个结构, //此时程序阻塞到该句柄中,等待句柄的进程结束或超时 if not CreateProcess(nil, pchar(cmdLine), nil, nil, false, CREATE_NEW_CONSOLE or NORMAL_PRIORITY_CLASS, nil, nil, sInfo, pInfo) then begin MsgDsp('WinExec调试失败!'); MessageBox(Application.handle, '指定程序启动失败!', '错误', MB_OK or MB_ICONSTOP); end else begin //等待指定句柄的进程结束或超时 WaitForSingleObject(pInfo.hProcess, INFINITE); GetExitCodeProcess(pInfo.hProcess, exitCode); MsgDsp('WinExec调试成功!'); end; end; procedure TFrmMain.btnClearClick(Sender: TObject); begin mmMsg.Clear; end; end.
运行效果如下:
总结
以上所述是小编给大家介绍的Delphi 调用外部程序并阻塞到外部程序中,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对好代码网网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!