6

我有一个有趣的(阅读:令人沮丧的)问题,从 C# WPF 应用程序启动控制台应用程序并重定向其标准输入和标准输出。

它大部分都在工作,但是一旦我开始重定向标准输入,我似乎最终没有从标准输出获得一些数据。

我会用一个例子来澄清。如果我没有在 STARTUPINFO 结构中设置 hStdInput,当我启动子进程时,我会收到以下信息:

MongoDB shell version: 2.2.0
connecting to: test
local:PRIMARY>

然而,一旦我设置了 hStdInput,我就会得到这个:

MongoDB shell version: 2.2.0
connecting to: test

我知道处理标准输出的 BackgroundWorker 仍在运行,因为如果我在标准输入上向进程发送某些内容,它会相应地响应。

use TestDB
switched to db TestDB

所以,这就是我创建过程的方式:

_processInfo = new ProcessInfo();

bool ok = false;

SECURITY_ATTRIBUTES sattr = new SECURITY_ATTRIBUTES();
sattr.bInheritHandle = 1;
unsafe
{
    sattr.lpSecurityDescriptor = null;
}
sattr.nLength = Marshal.SizeOf(sattr);

IntPtr hWrite;
ok = CreatePipe(out _hReadStdOut, out hWrite, ref sattr, 0);
ok = SetHandleInformation(_hReadStdOut, HANDLE_FLAGS.INHERIT, 0);
IntPtr hRead;
ok = CreatePipe(out hRead, out _hWriteStdIn, ref sattr, 0);
ok = SetHandleInformation(_hWriteStdIn, HANDLE_FLAGS.INHERIT, 0);

var startInfo = new StartupInfo
{
    dwFlags = 0x0001 | 0x0100,
    wShowWindow = 0,
    hStdOutput = hWrite,
    hStdError = hWrite,
    hStdInput = hRead // If this is IntPtr.Zero, I get everything from stdout
};

SECURITY_ATTRIBUTES pSec = new SECURITY_ATTRIBUTES();
pSec.nLength = Marshal.SizeOf(pSec);
SECURITY_ATTRIBUTES tSec = new SECURITY_ATTRIBUTES();
tSec.nLength = Marshal.SizeOf(tSec);

unsafe
{
    ok = CreateProcess(
        null,
        pathToExeAndArgs,
        ref pSec,
        ref tSec,
        true,
        0,
        IntPtr.Zero,
        null,
        ref startInfo,
        out _processInfo);
}

我在 DoWork 上有一个 BackgroundWorker 处理标准输出,它像这样读取管道:

success = ReadFile(
    _hReadStdOut,
    bufPtr,
    1024,
    &read,
    IntPtr.Zero);

我没有使用 .Net Process 类,因为在控制台应用程序发送换行符之前它没有从 stdout 获取数据,所以在那种情况下我也没有得到提示。

非常感谢您对此的任何帮助。

干杯。

4

1 回答 1

1

我怀疑以下内容可以解释您所观察到的情况:

  • 当您未定义hStdInput子进程时,将使用附加到控制台的标准输入设备。子进程检测到标准输入是交互式控制台设备并编写提示。
  • 当您定义hStdInput子进程时,检测到标准输入是管道,因此忽略了编写提示。毕竟,提示非交互式输入设备有什么意义呢?

子进程将用于GetFileType(GetStdHandle(STD_INPUT_HANDLE))检测连接到标准输入的设备类型。值FILE_TYPE_CHAR表示控制台。当您将管道附加到标准输入时,标准输入文件类型将为FILE_TYPE_PIPE.

我的结论是,一切都按设计和预期工作。

于 2012-12-08T16:06:14.560 回答