簡體   English   中英

背景工作者訪問UI的正確方法

[英]Background worker proper way to access UI

我不確定我是否正確地執行此操作,但是我使用了以下代碼(單擊button1,執行_DoWork)。 問題是:如何調用UI來獲取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);
            }

        }
    }

我應該使用后台工作人員嗎?

您可以通過RunWorkerAsync調用的參數將數據傳遞給worker,並通過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.
     }
  }

顧名思義,后台工作程序不會在UI線程上運行。 您只能在UI線程上訪問UI控件。 解決此問題的一種簡單方法是在需要新“對象”時保存所需的文本框屬性,然后將其傳遞給RunWorkerAsync。 此對象可用於e.Argument中的DoWork方法。

但是,在工作線程上顯示表單也存在問題。

您無法直接從BackgroundWorker訪問UI元素。要這樣做,您必須使用Dispatcher 從DependencyObject派生的WPF對象具有線程親和性 ,這意味着只有實例化它們的線程才能訪問其成員。

請查看下面的鏈接,看看代碼示例是否對您有所幫助

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

this.Dispather.Invoke((Action)delegate(){
    this.Close();
});

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM