创建一个cmd窗口并从C#应用程序写入它

Toi*_*wén 4 c# stdin cmd stdout

我正在为Rhino的Grasshopper开发一个C#组件.当我运行一些非常繁重的迭代分析时,我想将结果连续输出到cmd窗口,以确保分析实际上正在运行.

这是我试过的:

using System.Diagnostics;


Result results = new Result();
Process cmd = new Process();
cmd.StartInfo.FileName = "cmd.exe";
cmd.StartInfo.RedirectStandardInput = true;
cmd.StartInfo.RedirectStandardOutput = true;
cmd.StartInfo.CreateNoWindow = false;
cmd.StartInfo.UseShellExecute = false;
cmd.Start();

do {
    results = RunHeavyOperation(results);
    cmd.StandardInput.WriteLine("echo " + results.usefulInfo);
} while (!results.conditionForEnd);

cmd.WaitForExit();

Result RunHeavyOperation(Result previousResults) {
    Result res = doHeavyStuff(previousResults);
    return res;
}
Run Code Online (Sandbox Code Playgroud)

我意识到我错过了一部分,但它是什么?

Ren*_*ogt 6

您的方法是错误的:您当前没有写入控制台窗口.相反,您通过启动cmd.exe并写入该进程标准输入管道来创建进程.cmd.exe我不知道.这与通过键盘输入控制台不同,甚至可能产生奇怪的效果.
想象一下,您输出一个新的行字符,因此cmd.exe可能会尝试"执行"您之前输出的内容作为命令.

正确的方法是调用AllocConsole.通过此调用,您可以为流程创建一个控制台窗口,并通过它简单地使用它Console.WriteLine().

完成工作和记录后,最终需要关闭并再次释放此控制台FreeConsole.

所以导入这两个本机方法:

internal sealed class NativeMethods
{
    [DllImport("kernel32.dll")]
    public static extern bool AllocConsole();

    [DllImport("kernel32.dll")]
    public static extern bool FreeConsole();
}
Run Code Online (Sandbox Code Playgroud)

并在您的代码中使用它们:

NativeMethods.AllocConsole();

// start work
Console.WriteLine("log messages...");

// finished work

NativeMethods.FreeConsole();
Run Code Online (Sandbox Code Playgroud)

请注意,FreeConsole()关闭控制台窗口,因此所有日志消息都会丢失.并且控制台只有一个如此大的缓冲区,如果离开缓冲区,则无法滚动回较旧的消息.

因此,将日志消息简单地写入您可以在以后分析的文件中可能是一个更好的主意.