在Delphi 2009中如何重定向控制台输入(stin)和错误输出(sterr)?

3

我在互联网上尝试了几个示例,但它们都无法正常工作 - 脚本没有执行 - (可能是因为它们适用于早于Delphi 2009的unicode版本?)。

我需要运行一些Python脚本并传递参数给它们,例如:

python "..\Plugins\RunPlugin.py" -a login -u Test -p test

将输出捕获为字符串,错误信息捕获到其他位置。

这是我现在所拥有的:

procedure RunDosInMemo(DosApp:String; var OutData: String);
var
  SA: TSecurityAttributes;
  SI: TStartupInfo;
  PI: TProcessInformation;
  StdOutPipeRead, StdOutPipeWrite: THandle;
  WasOK: Boolean;
  Buffer: array[0..255] of Char;
  BytesRead: Cardinal;
  WorkDir: string;
  Handle: Boolean;
begin
  OutData := '';
  with SA do begin
    nLength := SizeOf(SA);
    bInheritHandle := True;
    lpSecurityDescriptor := nil;
  end;
  CreatePipe(StdOutPipeRead, StdOutPipeWrite, @SA, 0);
  try
    with SI do
    begin
      FillChar(SI, SizeOf(SI), 0);
      cb := SizeOf(SI);
      dwFlags := STARTF_USESHOWWINDOW or STARTF_USESTDHANDLES or CREATE_UNICODE_ENVIRONMENT;
      wShowWindow := SW_HIDE;
      hStdInput := GetStdHandle(STD_INPUT_HANDLE); // don't redirect stdin
      hStdOutput := StdOutPipeWrite;
      hStdError := StdOutPipeWrite;
    end;
    WorkDir := 'C:\';
    Handle := CreateProcess(nil, PChar(DosApp),
                            nil, nil, True, 0, nil,
                            PChar(WorkDir), SI, PI);
    CloseHandle(StdOutPipeWrite);
    if Handle then
    begin
      try
        repeat
          WasOK := ReadFile(StdOutPipeRead, Buffer, 255, BytesRead, nil);
          if BytesRead > 0 then
          begin
            Buffer[BytesRead] := #0;
            OutData := OutData + String(Buffer);
          end;
        until not WasOK or (BytesRead = 0);
        WaitForSingleObject(PI.hProcess, INFINITE);
      finally
        CloseHandle(PI.hThread);
        CloseHandle(PI.hProcess);
      end;
    end else begin
      raise Exception.Create('Failed to load python plugin');
    end;
  finally
    CloseHandle(StdOutPipeRead);
  end;
end;
2个回答

7
Create_Unicode_Environment是一个进程创建标志,用于CreateFiledwCreationFlags参数中。它不是TStartupInfo记录中使用的标志。如果您给API函数传递了它们不理解的标志值,则这些函数可能会失败;如果您给它们意味着您预期之外的标志值,则它们可能会执行奇怪的操作。
您声明了一个256个Char的缓冲区;请记住,Delphi 2009中的Char是一个2字节的Unicode类型。然后,您调用ReadFile并告诉它缓冲区的长度为255个字节,而不是实际值512。当文档表示一个值是字节数时,请将其视为使用SizeOf函数的提示。
由于ReadFile读取字节,因此最好将缓冲区数组声明为字节大小的元素数组,例如AnsiChar。这样,当您设置Buffer[BytesRead]时,您不会包含两倍于实际读取数据的数据。 CreateProcess的Unicode版本可能会修改其命令行参数。您必须确保传递给该参数的字符串具有1个引用计数。在调用CreateProcess之前,请调用UniqueString(DosApp)
当API函数失败时,您当然想知道原因。不要随便编造一个原因。使用提供的函数,例如Win32CheckRaiseLastOSError。至少调用GetLastError,就像MSDN告诉您的那样。当更具体的异常类型容易获得时,请不要抛出通用异常类型。

4

我不确定 WaitForSingleObject 是正确的方法... 我认为最好使用 GetExitCodeProcess(pi.hProcess, iExitCode) 循环,直到 iExitCode <> STILL_ACTIVE,然后在每次循环中检查数据。

代码如所写,在 Delphi 2007 下也无法运行,因此这不是 Delphi 2009 Unicode 的问题。

将您的内部循环更改为以下内容即可:

if Handle then
begin
  try
    repeat
      WasOK := ReadFile(StdOutPipeRead, Buffer, 255, BytesRead, nil);
      for ix := 0 to BytesRead-1 do
        begin
          OutData := OutData + AnsiChar(Buffer[ix]);
        end;
      GetExitCodeProcess(pi.hProcess,iExit);
    until (iExit <> STILL_ACTIVE);
  finally
    CloseHandle(PI.hThread);
    CloseHandle(PI.hProcess);
  end;

我对本地变量进行了以下更正/添加:

我对本地变量进行了以下更正/添加:

Buffer: array[0..255] of byte;
iExit : Cardinal;
IX : integer;

我将CloseHandle(StdOutPipeWrite)移动到了关闭StdOutPipeRead之前。

该段内容涉及it技术。

如果脚本以前没有运行过,那些更改不会使脚本运行。 "for"循环运行了一次太多; 修复它,将零分配到Buffer [BytesRead]中将不再起作用。通过从一开始就将缓冲区声明为AnsiChar数组来避免类型转换。你的WaitForSingleObject调用的目的是什么?你忽略了返回值。 - Rob Kennedy
waitforsingleobject调用是一个“占位符”,用于“消耗一些时间”。使用sleep也可以,但带有超时的waitforsingleobject只是表示等待,除非应用程序不再运行。我只是试图在循环中不占用CPU。 - skamradt
好的,我从别处复制了这段代码,我不擅长API开发,所以我很难理解其中的内容。谢谢你的帮助! - mamcx
1
延迟是不必要的。ReadFile函数只有在有数据可读或管道已断开时才会返回。 - Rob Kennedy

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接