Delphi 6-阅读控制台应用程序';在运行时显示输出
如何读取正在运行的控制台应用程序输出。我启动控制台应用程序,并希望在控制台应用程序打印输出时读取输出。这是怎么回事Delphi 6-阅读控制台应用程序';在运行时显示输出,delphi,console,Delphi,Console,如何读取正在运行的控制台应用程序输出。我启动控制台应用程序,并希望在控制台应用程序打印输出时读取输出。这是怎么回事 编辑:该链接指向此解决方案(为了清晰易读,对其进行了轻微的重构,并删除了对的使用): //该示例运行'chkdsk.exe c:\'并将输出显示为Memo1。 //在表格上写上一个TMemo(备忘录1)和一个T按钮(按钮1)。把这个 //按钮1的OnCLick事件过程中的代码: 过程TForm1.rundosinemo(DosApp:String;AMemo:TMemo); 常数
编辑:该链接指向此解决方案(为了清晰易读,对其进行了轻微的重构,并删除了对的使用):
//该示例运行'chkdsk.exe c:\'并将输出显示为Memo1。
//在表格上写上一个TMemo(备忘录1)和一个T按钮(按钮1)。把这个
//按钮1的OnCLick事件过程中的代码:
过程TForm1.rundosinemo(DosApp:String;AMemo:TMemo);
常数
ReadBuffer=2400;
变量
安全性:t安全属性;
读管,
书写体:THandle;
开始:TStartUpInfo;
ProcessInfo:TProcessInformation;
缓冲区:Pchar;
拜特斯拉德:德沃德;
通知:德沃德;
开始
Security.nlength:=SizeOf(TSecurityAttributes);
Security.binherithandle:=true;
Security.lpsecuritydescriptor:=nil;
如果Createpipe(ReadPipe,WritePipe,@Security,0),则
开始
缓冲区:=AllocMem(ReadBuffer+1);
FillChar(开始,Sizeof(开始),#0);
start.cb:=SizeOf(start);
start.hStdOutput:=writepe;
start.hStdInput:=读取管道;
start.dwFlags:=STARTF_USESTDHANDLES+STARTF_USESHOWWINDOW;
start.wShowWindow:=SW_HIDE;
如果CreateProcess(无,
PChar(DosApp),
@保安,,
@保安,,
是的,
正常优先级等级,
无
无
开始
ProcessInfo)然后
开始
重复
Apprunning:=WaitForSingleObject(ProcessInfo.hProcess,100);
Application.ProcessMessages;
直到(通知等待超时);
重复
字节读取:=0;
ReadFile(ReadPipe,缓冲区[0],
读取缓冲区,字节读取,无);
缓冲区[字节读取]:=#0;
OemToAnsi(缓冲区,缓冲区);
AMemo.Text:=AMemo.Text+字符串(缓冲区);
直到(字节读取<读取缓冲区);
结束;
FreeMem(缓冲区);
CloseHandle(ProcessInfo.hproces);
CloseHandle(ProcessInfo.hThread);
关闭手柄(读取管);
关闭手柄(WritePie);
结束;
结束;
程序TForm1.按钮1单击(发送方:TObject);
开始
rundosinemo('chkdsk.exe c:\',备忘录1);
结束;
我通常使用此端口FPC代码:
它包含一个控制外部程序的类(Lazarus用来调用cmdline编译器和其他程序的类)
这里有文档,但是delphi端口有点旧,所以上面的版本中可能不存在所有文档化的属性
仅对Marco answer的一个小补充,对使用TProcess单元进行了详细说明
我相信这是最简单的方法。祝你好运 对于那些关注链接的人来说,这是一个很好的建议,但是仅仅一个链接并不是一个很好的答案,请参见:@Mick:oops-当你在新答案中这样做时,我正在编辑旧的答案。如果有人认为我的编辑应该回滚,请随意这样做。不用担心。我当时删除了我的评论。这不会在它出现时读取输出,它只会在最后正确地读取所有内容。。。这不是我要找的。遗憾的是,这不是你要找的,但这段代码正是我要找的,谢谢。在最近的FPC版本中,一些修复一些最简单场景的简单过程被添加为“RunCommand()
// The example runs 'chkdsk.exe c:\' and displays the output to Memo1.
// Put a TMemo (Memo1) and a TButton (Button1) on your form. Put this
// code in the OnCLick event procedure for Button1:
procedure TForm1.RunDosInMemo(DosApp:String;AMemo:TMemo) ;
const
ReadBuffer = 2400;
var
Security : TSecurityAttributes;
ReadPipe,
WritePipe : THandle;
start : TStartUpInfo;
ProcessInfo : TProcessInformation;
Buffer : Pchar;
BytesRead : DWord;
Apprunning : DWord;
begin
Security.nlength := SizeOf(TSecurityAttributes) ;
Security.binherithandle := true;
Security.lpsecuritydescriptor := nil;
if Createpipe (ReadPipe, WritePipe, @Security, 0) then
begin
Buffer := AllocMem(ReadBuffer + 1) ;
FillChar(Start,Sizeof(Start),#0) ;
start.cb := SizeOf(start) ;
start.hStdOutput := WritePipe;
start.hStdInput := ReadPipe;
start.dwFlags := STARTF_USESTDHANDLES + STARTF_USESHOWWINDOW;
start.wShowWindow := SW_HIDE;
if CreateProcess(nil,
PChar(DosApp),
@Security,
@Security,
true,
NORMAL_PRIORITY_CLASS,
nil,
nil,
start,
ProcessInfo) then
begin
repeat
Apprunning := WaitForSingleObject(ProcessInfo.hProcess,100);
Application.ProcessMessages;
until (Apprunning <> WAIT_TIMEOUT) ;
repeat
BytesRead := 0;
ReadFile(ReadPipe,Buffer[0],
ReadBuffer,BytesRead,nil) ;
Buffer[BytesRead]:= #0;
OemToAnsi(Buffer,Buffer) ;
AMemo.Text := AMemo.text + String(Buffer) ;
until (BytesRead < ReadBuffer) ;
end;
FreeMem(Buffer) ;
CloseHandle(ProcessInfo.hProcess) ;
CloseHandle(ProcessInfo.hThread) ;
CloseHandle(ReadPipe) ;
CloseHandle(WritePipe) ;
end;
end;
procedure TForm1.Button1Click(Sender: TObject) ;
begin
RunDosInMemo('chkdsk.exe c:\', Memo1) ;
end;