后台工作线程访问UI的正确方式

8

我不确定我是否做得正确,但我正在使用以下代码(点击button1,执行_DoWork)。问题是:如何调用UI以获取textbox1和textbox2的值,因为它们不能在不同的线程上调用。我应该使用dispatchers吗?

    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

您可以通过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
如其名,后台工作者不在UI线程上运行。只有在UI线程上才能访问UI控件。解决此问题的一种简单方法是将所需的文本框属性保存到一个新的“对象”中,然后将其传递给RunWorkerAsync。这个对象可以在e.Argument中被DoWork方法使用。
但是,在工作线程上显示表单也会出现问题。

我知道,我对C#非常陌生,如果您知道更好的方法来实现这一点,那将不胜感激!我使用bgworker的原因是在执行sqlcommand时UI会冻结(以及当服务器不可用时)。 - Prisoner
UI卡顿不是很好吗?在用户登录之前,您不希望用户进行任何操作。但是,如果您想遵循您的模式,则可以在DoWork中执行SQL,并添加RunWorkerCompleted以进行UI呈现。 - Richard Schneider
4
以“在UI线程运行耗时操作导致UI冻结”意义上的UI冻结是不可接受的。 - Adam Lear

0

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

我还没有进展,访问textBox1和textBox2时出现错误。 - Prisoner

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接