Перенаправление вывода (stdout, stderr) дочернего процесса в окно вывода в Visual Studio

на данный момент я запускаю пакетный файл из своей программы на C# с помощью:

System.Diagnostics.Process.Start(@"DoSomeStuff.bat");

что я хотел бы сделать, это перенаправить вывод (stdout и stderr) этого дочернего процесса в окно вывода в Visual Studio (в частности, Visual C# Express 2008).

есть ли способ сделать это?

(дополнительно: так что это не все буферизовано, а затем выплюнуто в окно вывода, когда дочерний процесс завершится.)


(BTW: На данный момент я могу получить stdout (но не stderr) из родитель процесс появится в окне вывода, сделав мою программу " приложением Windows "вместо"консольного приложения". Это прерывается, если программа запускается за пределами Visual Studio, но в моем конкретном случае это нормально.)

4 ответов


process.StartInfo.CreateNoWindow = true;
process.StartInfo.UseShellExecute = false;
process.StartInfo.RedirectStandardOutput = true;
process.OutputDataReceived += (sender, args) => Console.WriteLine(args.Data);
process.Start();
process.BeginOutputReadLine();

process.WaitForExit();

идея Error, просто заменить Output в этих именах методов / свойств.


вариант этого работает для меня-опубликовать это сейчас, потому что я хотел бы найти его раньше. Обратите внимание, что это всего лишь фрагмент, извлеченный из реального кода, поэтому могут быть тривиальные ошибки.

метод основан на некотором коде MSDN. Что я не смог выяснить, как получить окно вывода для обновления "на лету". Он не обновляется до тех пор, пока эта задача не вернется.

// Set this to your output window Pane
private EnvDTE.OutputWindowPane _OutputPane = null;

// Methods to receive standard output and standard error

private static void StandardOutputReceiver(object sendingProcess, DataReceivedEventArgs outLine)
{
   // Receives the child process' standard output
   if (! string.IsNullOrEmpty(outLine.Data)) {
       if (_OutputPane != null)
           _OutputPane.Write(outLine.Data + Environment.NewLine);
   }
}

private static void StandardErrorReceiver(object sendingProcess, DataReceivedEventArgs errLine)
{
   // Receives the child process' standard error
   if (! string.IsNullOrEmpty(errLine.Data)) {
       if (_OutputPane != null)
           _OutputPane.Write("Error> " + errLine.Data + Environment.NewLine);
   }
}

// main code fragment
{
    // Start the new process
    ProcessStartInfo startInfo = new ProcessStartInfo(PROGRAM.EXE);
    startInfo.Arguments = COMMANDLINE;
    startInfo.WorkingDirectory = srcDir;
    startInfo.UseShellExecute = false;
    startInfo.RedirectStandardOutput = true;
    startInfo.RedirectStandardError = true;
    startInfo.CreateNoWindow = true;
    Process p = Process.Start(startInfo);
    p.OutputDataReceived += new DataReceivedEventHandler(StandardOutputReceiver);
    p.BeginOutputReadLine();
    p.ErrorDataReceived += new DataReceivedEventHandler(StandardErrorReceiver);
    p.BeginErrorReadLine();
    bool completed = p.WaitForExit(20000);
    if (!completed)
    {
        // do something here if it didn't finish in 20 seconds
    }
    p.Close();
}

здесь происходит то, что Visual Studio отображает вывод отладки из программы в окне вывода. То есть: если вы используете следа.WriteLine, он появится в окне вывода, из-за прослушивателя трассировки по умолчанию.

каким-то образом ваше приложение Windows Form (когда оно использует консоль.WriteLine; я предполагаю, что вы используете консоль.WriteLine) также записывает выходные данные отладки, и Visual Studio подхватывает это.

Он не будет делать то же самое для дочерних процессов, если вы явно не захватываете вывод и не перенаправляете его вместе с выводом.


вы рассматривали использование DefaultTraceListener ?

    //Create and add a new default trace listener.
    DefaultTraceListener defaultListener;
    defaultListener = new DefaultTraceListener();
    Trace.Listeners.Add(defaultListener);