16

我正在为 Windows 编写一个基本的 shell,我想知道是否有任何方法可以运行子进程 ( Process process),以便它使用当前的控制台窗口。我的意思是我不想重定向输入/输出;我希望该进程从当前控制台获取输入并将输出直接打印到同一个控制台窗口。

原因是我想允许这个子进程为输出设置控制台颜色,如果我重定向进程的标准输出就不会发生这种情况。另外,我目前使用的代码

while (!process.HasExited)
    process.StandardInput.WriteLine(Console.ReadLine());

将标准输入重定向到进程。但是,如果进程在输入后立即退出(例如,我键入“exit” + ENTER,进程退出),则此循环将再次运行,因此控制台正在等待用户的输入,该输入将永远不会使用由进程(即将退出)。

那么,很长的问题,我如何在当前控制台中运行一个进程,以便它可以设置控制台颜色并直接从控制台获取输入

编辑:以下是我的代码中与此问题相关的方法:

static int runExe(string exePath, params string[] args)
{
    ProcessStartInfo startInfo = new ProcessStartInfo(exePath, args)
    {
        ErrorDialog = false,
        UseShellExecute = false,
        CreateNoWindow = true,
        RedirectStandardInput = true,
        RedirectStandardOutput = true,
        RedirectStandardError = true,
        RedirectStandardInput = true
    };
    Process process = new Process() { StartInfo = startInfo };
    process.Start();
    ReadThreadState stdout = readThread(process.StandardOutput, false);
    ReadThreadState stderr = readThread(process.StandardError, true);
    while (!process.HasExited)
        process.StandardInput.WriteLine(Console.ReadLine());
    stdout.stop = stderr.stop = true;
    return process.ExitCode;
}
class ReadThreadState
{
    public bool stop;
}
private static ReadThreadState readThread(StreamReader reader, bool isError)
{
    ReadThreadState state = new ReadThreadState();
    new Thread(() =>
    {
        while (!state.stop)
        {
            int current;
            while ((current = reader.Read()) >= 0)
                if (isError)
                    writeError(((char)current).ToString(), ConsoleColor.Red);
                else
                    Console.Write((char)current);
        }
    }).Start();
    return state;
}
4

2 回答 2

23

您需要创建一个ProcessStartInfo并设置UseShellExecutefalse

var info = new ProcessStartInfo("program.exe", "arguments");
info.UseShellExecute = false;
var proc = Process.Start(info);
proc.WaitForExit();

这将在同一个控制台中启动您的程序。

使用上述技术的工作程序:

private static void Main(string[] args)
{
    Console.WriteLine("Starting program");
    var saveColor = Console.BackgroundColor;
    Console.BackgroundColor = ConsoleColor.Blue;
    var info = new ProcessStartInfo("cmd", "/c time");
    info.UseShellExecute = false;
    var proc = Process.Start(info);
    proc.WaitForExit();

    Console.BackgroundColor = saveColor;
    Console.WriteLine("Program exited");
    Console.ReadLine();
}

当您运行该程序时,它会启动一个新的 cmd.exe 副本并运行 time 命令,该命令要求输入。我这里只是以 cmd.exe 为例进行说明。任何从标准输入读取的程序都可以工作。另请注意,控制台颜色可以正常工作。

于 2013-07-19T18:09:47.423 回答
2

Jim Mischel's answer works like a charm. I'm moving some batch file processing into C# and that works great. You may find this small utility method handy. Just takes a command line, and runs it just like CALL in a batch file. Just stuff it in a utility class:

public static void RunCmd(string command) {
    Process.Start(new ProcessStartInfo("cmd.exe", "/c " + command) {
        UseShellExecute = false
    }).WaitForExit();            
}    
于 2014-03-19T23:20:25.503 回答