Асинхронный вывод из приложения командной строки? - PullRequest
0 голосов
/ 30 ноября 2011

Я сделал приложение для обновления устройства, которое первоначально использовало командную строку, поэтому я сделал это в пользовательском интерфейсе WPF.

я написал этот код для запуска процесса и обработки полученных данных, а также для отлова ошибок

System.Diagnostics.Process StarterBackup_X64 = new System.Diagnostics.Process();
                                        StarterBackup_X64.StartInfo.FileName = X64;
                                        var Starter = StarterBackup_X64;
                                        Starter.StartInfo.Arguments = string.Concat("/iu \"", textBox1.Text, " " + textBox2.Text, " " + textBox3.Text, " " + textBox4.Text, " " + textBox5.Text, " " + textBox6.Text, " " + textBox7.Text, " " + textBox8.Text, " " + textBox9.Text, " " + textBox10.Text, " " + textBox11.Text, " " + textBox12.Text, " " + textBox13.Text, " " + textBox14.Text, " " + textBox15.Text, " " + textBox16.Text, " " + textBox17.Text, " " + textBox18.Text, " " + textBox19.Text, " " + textBox20.Text, "\" /enablebackup");
                                        //Starter.StartInfo.Arguments = string.Concat("/iu\"", textBox1.Text + textBox2.Text + textBox3.Text + textBox4.Text + textBox5.Text + textBox6.Text + textBox7.Text + textBox8.Text + textBox9.Text + textBox10.Text + textBox11.Text + textBox12.Text + textBox13.Text + textBox14.Text + textBox15.Text + textBox16.Text + textBox17.Text + textBox18.Text + textBox19.Text + textBox20.Text, "\" /enablebackup");
                                        StarterBackup_X64.StartInfo.CreateNoWindow = true;
                                        StarterBackup_X64.StartInfo.RedirectStandardOutput = true;
                                        StarterBackup_X64.StartInfo.UseShellExecute = false;
    StarterBackup_X64.OutputDataReceived += new       DataReceivedEventHandler(StarterBackup_X64_OutputDataReceived);
                                        StarterBackup_X64.Start();
    StarterBackup_X64.BeginOutputReadLine();
    installUpdateButton.Content = "Updating device.....";
    installUpdateButton.IsEnabled = false;
    restoreDeviceButton.IsEnabled = false;

    string UpdaterLog = outputTextBox.Text;

    if (UpdaterLog.ToString().IndexOf("no devices were found") > -1)
    {
              WPECore.ResTable.DeviceNotFound();
    }

    else if (UpdaterLog.ToString().IndexOf("error:") > -1)
    {
              WPECore.ResTable.UpdateWPCrashed();
    }

    else if (UpdaterLog.ToString().IndexOf("error message:") > -1)
    {
               WPECore.ResTable.UpdateWPCrashed();
    }

    void StarterBackup_X64_OutputDataReceived(object sender, DataReceivedEventArgs e)
    {
        outputTextBox.Dispatcher.BeginInvoke(new Action(() => { outputTextBox.Text += e.Data;  }), null);
        SplitTextIntoLines(outputTextBox.Text, 1);


    }

и приложение работает, но вывод не в аккуратном виде, поэтому я попытался написать эту функцию

        public static string[] SplitTextIntoLines(string title, int width)
    {
        System.Collections.Generic.List<System.String> list;
        System.Text.StringBuilder stringBuilder;
        string str;
        string[] arrstr1;
        int i;
        list = new System.Collections.Generic.List<System.String>();
        stringBuilder = new System.Text.StringBuilder();
        arrstr1 = title.Split(new char[] {
            ' '});
        i = 0;
        while (i < arrstr1.Length)
        {
            str = arrstr1[i];
            if (((stringBuilder.Length + str.Length) + 1) <= width)
            {
                stringBuilder.Append(' ');
                stringBuilder.Append(str);
            }
            else
            {
                list.Add(stringBuilder.ToString().Trim());
                stringBuilder = new System.Text.StringBuilder(str);
            }
            i++;
        }
        list.Add(stringBuilder.ToString().Trim());
        return list.ToArray();
    }

и я добавил SplitTextIntoLines () в обработчик событий

но это привело к исключению InvalidOperationException: вызывающий поток не может получить доступ к этому объекту, так как он принадлежит другому потоку.

так ,. Что я должен сделать, чтобы вывод этого приложения был аккуратным?

1 Ответ

1 голос
/ 30 ноября 2011

Используйте следующее:

string outputText;
    void StarterBackup_X64_OutputDataReceived(object sender, DataReceivedEventArgs e)
    {
            outputTextBox.Dispatcher.Invoke(new Action(() => { outputTextBox.Text += e.Data; outputText = outputTextBox.Text; }), null);
            SplitTextIntoLines(outputText, 1);
    }
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...