Перенаправление вывода (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);