BackgroundWorker线程必须是STA

3

我有一个BackgroundWorker,在BackgroundWorker _DoWork中调用一个函数来执行长时间的处理,当函数发生错误时,我会提示一个自定义消息框:

 WPFMessageBoxResult result = WPFMessageBox.Show("Activation Fail", "Error!!", WPFMessageBoxButtons.OK, WPFMessageBoxImage.Error);

以下异常发生在WPFMessageBoxResult类中:
The calling thread must be STA, because many UI components require this. 

感谢您的选择。
4个回答

4

您不应尝试从后台线程与任何UI组件进行交互。

一种方法是在doWork方法中捕获异常,并将其分配给backgroundworker的结果属性,如果您没有将结果用于其他任何内容,则可以检查该结果是否为异常类型或非空,然后在backgroundWorker_completed事件中进行检查。

BackgroundWorker_DoWork(sender, )
{
    try
    {
       // do work        
    }
    catch (Exception ex)
    {
         BackgroundWorker w = sender as BackgroundWorker;
         if (w != null)
             w.Result = ex;
    }
}

然后。
BackgroundWorker_Completed()
{
    if (s.Result != null && e.Result is Exception)
    {
       Exception ex = e.Result as Exception;
       // do something with ex
    }
}

4

通常情况下,在 Winforms/WPF 中,如果你需要从一个长时间运行的任务中与 UI 交互,你需要使用 Invoke() 方法来切换到 UI 线程。你可以在任何属于 UI 的对象上调用 Invoke() 方法,但是请确保在调用 Invoke() 方法时只执行尽可能少的代码,因为该代码将在 UI 线程上被执行,如果执行时间过长,将会导致 UI 阻塞/挂起。

public void BackgroundWorkerMethod()
{
    try
    {
        // do work
    }
    catch (Exception e)
    {
        uiObject.Invoke(new Action(() => {
            // now you are on the UI thread
            Message.ShowMessage(e.Message);
        });
    }
}

3

您必须使用此方法

 void BGW_DoWork(object sender, DoWorkEventArgs e)
    {
        try
        {
            Dispatcher.BeginInvoke(new Action(() =>
            {
                 Button btn = new Button();
                 btn.Width = 100;
                 btn.Height = 50;
                 btn.Content = "Test";
                 myG.Children.Add(btn);
            }
            ));
        }
            catch (Exception ex)
        {
            MessageBox.Show(ex.Message);
        }

    }

3

您的后台线程只是一个工作线程,而不是用户界面线程。WPF和WinForms都要求执行用户界面操作的线程标记为STA(单线程公寓),因为用户界面代码不是线程安全的。它们还要求您添加消息泵,以便分派窗口消息。

我建议您不要在工作线程中显示消息框,而是向主用户界面线程发送消息,并让该线程显示消息框。为此,您应该将Dispatcher的引用从主UI线程传递到工作线程中。然后使用Dispatcher.BeginInvoke请求在主线程上回调委托。

或者您可以等待后台线程完成,然后检查结果并向用户显示适当的答案。无论哪种方式,工作线程都不应直接与用户界面交互。


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