Я разрабатываю приложение Windows Forms, которое требует от меня вызова отдельной программы для выполнения задачи. Программа представляет собой консольное приложение, и мне нужно перенаправить стандартный вывод с консоли на TextBox в моей программе.
У меня нет проблем с выполнением программы из моего приложения, но я не знаю, как перенаправить вывод в свое приложение. Мне нужно захватить вывод, пока программа работает с использованием событий.
Консольная программа не предназначена для остановки работы до тех пор, пока мое приложение не остановится, а текст не будет постоянно меняться с произвольными интервалами. Я пытаюсь просто перехватить вывод консоли, чтобы запустить обработчик событий, который затем можно использовать для обновления TextBox.
Я использую C# для кодирования программы и использую платформу .NET для разработки. Исходное приложение не является программой .NET.
Обновлено: Вот пример кода того, что я пытаюсь сделать. В своем последнем приложении я заменю Console.WriteLine кодом для обновления TextBox. Я попытался установить точку останова в своем обработчике событий, но она даже не достигнута.
void Method()
{
var p = new Process();
var path = @"C:\ConsoleApp.exe";
p.StartInfo.FileName = path;
p.StartInfo.UseShellExecute = false;
p.OutputDataReceived += p_OutputDataReceived;
p.Start();
}
static void p_OutputDataReceived(object sender, DataReceivedEventArgs e)
{
Console.WriteLine(">>> {0}", e.Data);
}





См. Как запустить файл .EXE из C# и получить (передать) результаты из окон DOS обратно в C#
Не совсем, мне нужно захватить текст, когда он записывается в стандартный вывод.
Если вы перейдете по ссылке в сообщении, вы увидите здесь код: social.msdn.microsoft.com/forums/en-US/csharpgeneral/thread/…
Да, но этот код запускается только один раз. Я обновил свой вопрос, чтобы уточнить, что я пытаюсь сделать.
Это работает для меня:
void RunWithRedirect(string cmdPath)
{
var proc = new Process();
proc.StartInfo.FileName = cmdPath;
// set up output redirection
proc.StartInfo.RedirectStandardOutput = true;
proc.StartInfo.RedirectStandardError = true;
proc.EnableRaisingEvents = true;
proc.StartInfo.CreateNoWindow = true;
// see below for output handler
proc.ErrorDataReceived += proc_DataReceived;
proc.OutputDataReceived += proc_DataReceived;
proc.Start();
proc.BeginErrorReadLine();
proc.BeginOutputReadLine();
proc.WaitForExit();
}
void proc_DataReceived(object sender, DataReceivedEventArgs e)
{
// output will be in string e.Data
}
Я обнаружил, что иногда (при выходе из программы) e.Data имеет значение null, поэтому вам нужно проверить, если e.Data! = Null!
+1 Я впервые увидел что-то в этом роде в EditPlus. Теперь я могу делать это в моих .NET-программах. Спасибо чувак!
Разве нам не нужно настраивать UseShellExecute = false для перенаправления вывода?
Спасибо, Марк, однако использование этого со скомпилированным сценарием Ruby в качестве исполняемого файла с выводом только команды не совсем работает так, как задумано. Во-первых, как упоминает Джон на все руки, вам необходимо иметь proc.StartInfo.UseShellExecute = false; ... тогда вы можете все данные, отправленные через BeginOutputReadLine (), сразу после завершения программы. Я чувствую, что ReadLine буквально хочет строк вместо текста, поэтому, если коды возврата / новой строки, отправленные исполняемым файлом, не соответствуют среде, он не будет перехватывать какие-либо данные в OutputDataReceived, пока приложение не завершит работу.
Если вы используете этот код в производстве, помните, что Process является одноразовым, а также не забудьте отсоединить обработчики событий (proc.ErrorDataReceived -= proc_DataReceived и т. д.).
Вы можете использовать следующий код
MemoryStream mem = new MemoryStream(1000);
StreamWriter writer = new StreamWriter(mem);
Console.SetOut(writer);
Assembly assembly = Assembly.LoadFrom(@"C:\ConsoleApp.exe");
assembly.EntryPoint.Invoke(null, null);
writer.Close();
string s = Encoding.Default.GetString(mem.ToArray());
mem.Close();
Если «C: \ ConsoleApp.exe» не является приложением .Net, это не запустится!
Я добавил несколько вспомогательных методов в Платформа O2 (проект с открытым исходным кодом), которые позволяют легко создавать сценарии взаимодействия с другим процессом через вывод и ввод консоли (см. http://code.google.com/p/o2platform/source/browse/trunk/O2_Scripts/APIs/Windows/CmdExe/CmdExeAPI.cs).
Также вам может быть полезен API, который позволяет просматривать консольный вывод текущего процесса (в существующем элементе управления или всплывающем окне). См. Это сообщение в блоге для получения дополнительных сведений: http://o2platform.wordpress.com/2011/11/26/api_consoleout-cs-inprocess-capture-of-the-console-output/ (этот блог также содержит сведения о том, как использовать вывод консоли новых процессов)
Спасибо Марку Максхэму за его ответ, который сэкономил мне время!
Как заметил Джон на все руки, для UseShellExecute необходимо установить значение false, чтобы перенаправить потоки ввода-вывода, в противном случае вызов Start() выдает InvalidOperationException.
Вот моя модификация кода, где txtOut - это текстовое поле WPF только для чтения
void RunWithRedirect(string cmdargs)
{
var proc = new Process()
{
StartInfo = new ProcessStartInfo("cmd.exe", "/k " + cmdargs)
{
RedirectStandardOutput = true,
RedirectStandardError = true,
CreateNoWindow = true
},
EnableRaisingEvents = true
};
// see below for output handler
proc.ErrorDataReceived += proc_DataReceived;
proc.OutputDataReceived += proc_DataReceived;
proc.Start();
proc.BeginErrorReadLine();
proc.BeginOutputReadLine();
proc.WaitForExit();
}
void proc_DataReceived(object sender, DataReceivedEventArgs e)
{
if (e.Data != null)
Dispatcher.BeginInvoke(new Action( () => txtOut.Text += (Environment.NewLine + e.Data) ));
}
Измените исходный ответ вместо публикации собственного, чтобы автор получил оценку (y)
Где
InputDataReceived?