3

我刚刚开始处理 STA/MTA 问题,因此为问题的简单性道歉。我在这里的阶梯底部找不到我能真正理解的答案。

我正在为另一款软件编写插件,并在工作线程中发现我需要创建一些 UI 元素。我知道我不能从工作线程内部执行此操作,因为它不是 STA 线程,并且我需要返回主(或只是另一个?)STA 线程来创建 UI 元素。一些澄清将有很大帮助。

  1. 是否所有 STA 线程都具有相同的“权限”,即如果主线程是 STA 并创建一个窗口,则向其添加一些 UI 元素。然后产生另一个 STA 线程,第二个线程同样创建一些 UI 元素,它们是否在同一个“空间”中执行(糟糕的词选择,但我不知道还能使用什么)并且可以访问彼此的 UI 元素不会造成死亡和破坏?或者我是否需要明确地跳回主/原始 STA 线程并且只从那个(不仅仅是任何)STA 线程创建 UI 元素?

  2. 如果是这种情况(只允许 1 个 STA 线程制作 UI 元素),我该如何正确地做到这一点?我看过很多与此相关的帖子,但由于某种原因,我不太了解发生了什么,并且希望得到一个真正简单的答案。

请不要'这是一种很酷的巧妙方法......'我只需要在执行点的简单方法,如果这是必要的,我需要一些 UI 元素跳回主 STA 线程。

如果没有必要,那么我只会将该工作线程设为 STA 线程并继续进行,这样公平吗?还是我在招惹灾难?

4

2 回答 2

3

如果主线程是 STA 并创建一个 Window,则向其添加一些 UI 元素。然后产生另一个 STA 线程,第二个线程同样创建一些 UI 元素,它们是否在同一个“空间” [snip...]中进行,并且可以访问彼此的 UI 元素而不会造成死亡和破坏?

如果线程 A 和 B 都是 STA,那么它们可以各自创建和更新自己UI 元素,但不能互相创建和更新。任何其他想要影响 UI 的线程都必须使用一种BeginInvoke样式方法来请求相应的线程进行更新。

如果没有必要,那么我只会将该工作线程设为 STA 线程并继续进行,这样公平吗?还是我在招惹灾难?

如果工作线程已设置为 MTA 并已初始化,则您可能无法使工作线程成为 STA 线程。您可能需要创建一个新线程。

你怎么做呢?似乎您想System.Windows.*为您的 UI 使用 WPF ( ) - 所以如果您“插入”的应用程序也在使用 WPF,您应该能够访问它并重新使用它的 UI 线程。如果没有,您可以创建一个新线程,并Application在其上创建一个新线程并调用Run. 这应该为您设置一个调度程序。

像这样的东西(从我在其他地方的一些工作代码中复制的伪代码排序)

Dispatcher dispatcher = null; // we 'get to' the UI thread via the dispatcher

if(Application.Current) { // re use an existing application's UI thread
    dispatcher = Application.Current.Dispatcher;
} else {
    var threadReadyEvent = new ManualResetEvent(false);

    var uiThread = new Thread(() => {
        Thread.CurrentThread.SetApartmentState(ApartmentState.STA);

        var application = new Application();
        application.Startup += (sender, args) => {
            dispatcher = application.Dispatcher;
            threadReadyEvent.Set();
        };

        // apps have to have a "main window" - but we don't want one, so make a stub
        var stubWindow = new Window {
            Width = 1, Height = 1, 
            ShowInTaskbar = false, AllowsTransparency = true,
            Background = Brushes.Transparent, WindowStyle = WindowStyle.None
        };
        application.Run(stubWindow);
    }){ IsBackground = true };

    uiThread.Start();
    threadReadyEvent.WaitOne();
    threadReadyEvent.Dispose();
}

dispatcher.Invoke(() => {
    // ask the UI thread to do something and block until it finishes
});

dispatcher.BeginInvoke(() => {
    // ask the UI thread to do something asynchronously
});

等等

于 2014-01-07T01:46:48.663 回答
2
  1. 如果一个线程创建一个控件。只有这个特定的线程可以与之交互,即使还有其他 STA 线程。
  2. 在 WinForms 中,您将调用控件上的方法:Control.Invoke。在 WPF 中,您有调度程序来执行此操作:Dispatcher.Invoke

赢表格:

form1.Invoke(/* a delegate for your operation */)

WPF:

window1.Dispatcher.Invoke(/* a delegate for your operation */)

您所做的不是更改“单一公寓”中的对象,而是要求(调用)控制它的 STA 线程(您调用的委托)为您执行此操作。您还必须BeginInvoke异步执行此操作。

于 2014-01-06T23:12:58.270 回答