Перенаправить вывод консоли в текстовое поле в отдельной программе
Вопрос
Я разрабатываю приложение Windows Forms, которое требует от меня вызова отдельной программы для выполнения задачи.Программа представляет собой консольное приложение, и мне нужно перенаправить стандартный вывод с консоли в текстовое поле в моей программе.
У меня нет проблем с выполнением программы из моего приложения, но я не знаю, как перенаправить выходные данные в мое приложение.Мне нужно записывать выходные данные во время работы программы, используя события.
Консольная программа не должна прекращать работу до тех пор, пока мое приложение не остановится и текст не будет постоянно меняться через случайные промежутки времени.То, что я пытаюсь сделать, это просто подключить вывод из консоли для запуска обработчика событий, который затем можно использовать для обновления текстового поля.
Я использую C # для написания кода программы и использую .NET framework для разработки.Исходное приложение не является программой .NET.
Редактировать:Вот пример кода того, что я пытаюсь сделать.В моем последнем приложении я заменю Console.WriteLine кодом для обновления текстового поля.Я попытался установить точку останова в моем обработчике событий, но она даже не достигнута.
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);
}
Решение
Это работает для меня:
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
}
Другие советы
Вы можете использовать следующий код
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();
Я добавил ряд вспомогательных методов к Платформа 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) ));
}