我有一个有趣的(读取:令人沮丧的)问题从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获取数据,所以在这种情况下我也没有得到提示。
对此的任何帮助都非常感激。
干杯。
答案 0 :(得分:1)
我怀疑以下内容解释了您所观察到的内容:
hStdInput
,子进程将使用附加到控制台的标准输入设备。子进程检测到标准输入是交互式控制台设备并写入提示。hStdInput
时,子进程检测到标准输入是管道,因此忽略了写入提示。毕竟,提示非交互式输入设备的重点是什么?子进程将使用GetFileType(GetStdHandle(STD_INPUT_HANDLE))
来检测附加到标准输入的设备类型。值FILE_TYPE_CHAR
表示控制台。将管道连接到标准输入时,标准输入文件类型将为FILE_TYPE_PIPE
。
我的结论是,一切都按设计和预期运作。