Фоновый рабочий правильный способ доступа к пользовательскому интерфейсу - PullRequest
8 голосов
/ 22 февраля 2011

Я не уверен, правильно ли я это делаю, но у меня есть следующий код, который я использую (нажмите кнопку 1, выполните _DoWork).Вопрос заключается в следующем: как вызвать пользовательский интерфейс, чтобы получить значения textbox1 и textbox2, поскольку они не могут быть вызваны, поскольку они находятся в другом потоке.Должен ли я использовать диспетчеров?

    private void button1_Click(object sender, RoutedEventArgs e)
    {
        if (textBox1.Text == "")
        {
            MessageBox.Show("Please enter a username and password", "Error", MessageBoxButton.OK, MessageBoxImage.Warning);
        }
        else
        {
            bw.DoWork += new DoWorkEventHandler(bw_DoWork);
            bw.RunWorkerAsync();
        }
    }

    private void bw_DoWork(object sender, DoWorkEventArgs e)
    {
        Console.WriteLine("asd");
        UserManagement um = new UserManagement(sm.GetServerConnectionString());
        if (um.AuthUser(textBox1.Text, textBox2.Password))
        {
            MainWindow mw = new MainWindow();
            mw.Show();
            this.Close();
        }
        else
        {
            if (um.Timeout)
            {
                MessageBox.Show("Could not connect to server, please check your configuration", "Error", MessageBoxButton.OK, MessageBoxImage.Error);
            }
            else
            {
                MessageBox.Show("Incorrect username or password", "Error", MessageBoxButton.OK, MessageBoxImage.Error);
            }

        }
    }

Должен ли я использовать фоновый работник?

Ответы [ 4 ]

13 голосов
/ 22 февраля 2011

Вы можете передавать данные в рабочий через аргумент вызова RunWorkerAsync и передавать данные через DoWorkEventArgs.Result ...

  class AuthUserData
  {
    public string Name;
    public string Password;
  }

  private void button1_Click(object sender, EventArgs e)
  {
     var authData = new AuthUserData() { Name = textBox1.Text, Password = textBox2.Text };
     worker.RunWorkerAsync(authData);
  }

  void worker_DoWork(object sender, DoWorkEventArgs e)
  {
     // On the worker thread...cannot make UI calls from here.
     var authData = (AuthUserData)e.Argument;
     UserManagement um = new UserManagement(sm.GetServerConnectionString());
     e.Result = um;
     e.Cancel = um.AuthUser(textBox1.Text, textBox2.Password));
  }

  void worker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
  {
     // Back on the UI thread....UI calls are cool again.
     var result = (UserManagement)e.Result;
     if (e.Cancelled)
     {
        // Do stuff if UserManagement.AuthUser succeeded.
     }
     else
     {
        // Do stuff if UserManagement.AuthUser failed.
     }
  }
3 голосов
/ 22 февраля 2011

Как видно из названия, фоновый рабочий не работает в потоке пользовательского интерфейса. Вы можете получить доступ только к элементам управления пользовательского интерфейса, находясь в потоке пользовательского интерфейса. Простой способ обойти эту проблему - сохранить нужные свойства текстового поля в новом «объекте» и затем передать его в RunWorkerAsync. Этот объект доступен для вашего метода DoWork в e.Argument.

Однако у вас также есть проблема с отображением формы в рабочем потоке.

0 голосов
/ 22 февраля 2011

Вы не можете получить доступ к элементам пользовательского интерфейса напрямую из BackgroundWorker . Для этого вам нужно использовать Диспетчер . Объекты WPF, производные от DependencyObject, имеют сходство потоков , что означает, что только поток, который их создает, может получить доступ к своим членам.

Проверьте ссылку ниже и посмотрите, поможет ли вам пример кода

http://social.msdn.microsoft.com/Forums/en/wpf/thread/4858bcaf-1cb2-410b-989a-18b874ffa458

0 голосов
/ 22 февраля 2011
this.Dispather.Invoke((Action)delegate(){
    this.Close();
});
...