你能用你自己的话解释一下STA和MTA吗?

另外,什么是公寓线程,它们只与COM有关吗?如果有,为什么?


当前回答

这篇文章很清楚地解释了STA和MTA。

了解COM公寓,第一部分 理解COM公寓,第二部分

关于什么是Apartment的要点:

An apartment is a concurrency boundary; it’s an imaginary box drawn around objects and client threads that separates COM clients and COM objects that have incompatible threading characteristics. Every thread that uses COM, and every object that those threads create, is assigned to an apartment. When a thread calls COM’s CoInitialize or CoInitializeEx function, that thread is placed in an apartment. And when an object is created, it too is placed in an apartment. Whenever it creates a new apartment, COM allocates an apartment object on the heap and initializes it with important information such as the apartment ID and apartment type. When it assigns a thread to an apartment, COM records the address of the corresponding apartment object in thread-local storage (TLS).

其他回答

我发现现有的解释太官样文章了。下面是我的解释:

STA: If a thread creates a COM object that's set to STA (when calling CoCreateXXX you can pass a flag that sets the COM object to STA mode), then only this thread can access this COM object (that's what STA means - Single Threaded Apartment), other thread trying to call methods on this COM object is under the hood silently turned into delivering messages to the thread that creates(owns) the COM object. This is very much like the fact that only the thread that created a UI control can access it directly. And this mechanism is meant to prevent complicated lock/unlock operations.

MTA: 如果一个线程创建了一个设置为MTA的COM对象,那么几乎每个线程都可以直接调用该对象上的方法。

这差不多就是要点了。虽然从技术上讲,还有一些细节我没有提到,比如在“STA”段落中,创造者线程本身必须是STA。但这几乎是你了解STA/MTA/NA所需要知道的全部内容。

这完全取决于如何处理对对象的调用,以及它们需要多少保护。COM对象可以要求运行时保护它们不被多个线程同时调用;那些没有调用的线程可以从不同的线程并发调用,所以它们必须保护自己的数据。

此外,如果从用户界面线程进行调用,运行时还需要防止COM对象调用阻塞用户界面。

公寓是对象居住的地方,它们包含一个或多个线程。公寓定义了打电话时发生的事情。对公寓中对象的调用将在该公寓中的任何线程上接收和处理,除非已经在正确公寓中的线程的调用由自身处理(即对对象的直接调用)。

线程可以在单线程单元中(在这种情况下,它们是该单元中唯一的线程),也可以在多线程单元中。它们在线程初始化该线程的COM时指定哪个线程。

STA主要用于与用户界面的兼容性,用户界面绑定到特定的线程。STA通过接收到隐藏窗口的窗口消息来接收要处理的调用通知;当它进行出站调用时,它启动一个模态消息循环,以防止其他窗口消息被处理。您可以指定要调用的消息筛选器,以便应用程序可以响应其他消息。

相比之下,所有MTA线程共享一个进程的MTA。如果没有可用线程,COM可能会启动一个新的工作线程来处理传入的调用,直到池的限制。发出出站调用的线程只是阻塞。

为了简单起见,我们将只考虑在dll中实现的对象,这些对象通过为类的键设置ThreadingModel值,在注册表中通告它们支持什么。有四种选择:

Main thread (ThreadingModel value not present). The object is created on the host's main UI thread, and all calls are marshalled to that thread. The class factory will only be called on that thread. Apartment. This indicates that the class can run on any single-threaded-mode thread. If the thread that creates it is an STA thread, the object will run on that thread, otherwise it will be created in the main STA - if no main STA exists, an STA thread will be created for it. (This means MTA threads that create Apartment objects will be marshalling all calls to a different thread.) The class factory can be called concurrently by multiple STA threads so it must protect its internal data against this. Free. This indicates a class designed to run in the MTA. It will always load in the MTA, even if created by an STA thread, which again means the STA thread's calls will be marshalled. This is because a Free object is generally written with the expectation that it can block. Both. These classes are flexible and load in whichever apartment they're created from. They must be written to fit both sets of requirements, however: they must protect their internal state against concurrent calls, in case they're loaded in the MTA, but must not block, in case they're loaded in an STA.

在. net框架中,基本上只要在创建UI的任何线程上使用[STAThread]即可。工作线程应该使用MTA,除非它们打算使用带有公寓标记的COM组件,在这种情况下,如果从多个线程调用相同的组件(因为每个线程都必须依次等待组件),则使用STA来避免编组开销和可伸缩性问题。如果每个线程使用一个单独的COM对象,无论组件是在STA还是MTA中,都要容易得多。

COM线程模型称为“单元”模型,其中初始化的COM对象的执行上下文与单个线程(单线程单元)或多个线程(多线程单元)相关联。在这个模型中,COM对象一旦在一个单元中初始化,就在其运行期间是该单元的一部分。

STA模型用于非线程安全的COM对象。这意味着它们不处理自己的同步。它的一个常见用途是UI组件。因此,如果另一个线程需要与对象交互(例如按表单中的按钮),则消息将被编组到STA线程上。windows窗体消息泵送系统就是一个例子。

如果COM对象可以处理自己的同步,那么可以使用MTA模型,其中允许多个线程与对象交互,而不需要编组调用。

调用COM对象dll的代码(例如,读取私有数据文件)可能在用户界面中正常工作,但却神秘地挂在服务上。原因是。net 2.0的用户界面假设是STA(线程安全的),而服务假设是MTA((在此之前,服务假设是STA)。必须为服务中的每个COM调用创建一个STA线程会增加大量开销。

根据我的理解,“Apartment”是用来保护COM对象不受多线程问题的影响。

If a COM object is not thread-safe, it should declare it as a STA object. Then only the thread who creates it can access it. The creation thread should declare itself as a STA thread. Under the hood, the thread stores the STA information in its TLS(Thread Local Storage). We call this behavior as that the thread enters a STA apartment. When other threads want to access this COM object, it should marshal the access to the creation thread. Basically, the creation thread uses messages mechanism to process the in-bound calls.

如果一个COM对象是线程安全的,它应该声明为一个MTA对象。MTA对象可以被多线程访问。