C#console应用程序stdin / stdout重定向

时间:2012-12-08 14:11:41

标签: c# console pinvoke stdout

我有一个有趣的(读取:令人沮丧的)问题从C#WPF应用程序启动控制台应用程序并重定向其stdin和stdout。

它主要是启动和工作但我似乎最终在我开始重定向stdin时没有从stdout获取一些数据。

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

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

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

MongoDB shell version: 2.2.0
connecting to: test

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

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获取数据,所以在这种情况下我也没有得到提示。

对此的任何帮助都非常感激。

干杯。

1 个答案:

答案 0 :(得分:1)

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

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

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

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