多线程调用COM组件的体会(CoInitialize)

调用任何COM组件之前,你必须首先初始化COM套件环境,即调用CoInitialize或CoInitializeEx。COM套件环境在线程的生存周期内有效,线程退出前需要调用CoUninitialize释放COM套件。

 

所谓COM套件,实际上是微软为了方便大家理解而起的一个名字,不过个人认为改名词很难理解。COM套件只指COM组件运行时的环境,其中包括COM组件的数据、变量、线程调度方式。

 

COM套件分为两种模式,单线程套件(STA)和多线程套件(MTA)。不要单从字面上理解,例如:STA并非只能用于单线程的程序,多线程程序依然可以使用。下面列出两种套件模式的区别。



套件类型

说明

性能

兼容性

常见错误
STA

单线程套间,一个进程内所有COM组件都运行在主STA中,主STA就是第一个调用CoInitialize函数的线程。

也就是说,即使你拥有多线程程序,但在不通线程同时操作COM组件的时候,COM组件会通过Windows消息、Event同步对象之类的机制把调用转换到主STA执行,而主STA通常对应应用程序的主线程。这样对STA套件内的任何COM组件操作,实际上是单线程操作,COM组件不必关心线程同步的细节,因为根本没有必要进行线程同步。

如果一个COM组件是MTA的,可以安全的运行与STA套件中。

 

由于STA套件所有的COM组件代码都运行于主STA(第一个调用CoInitialize函数的线程),如果你的主线程没有调用CoInitialize,那么第一个调用CoInitialize的工作线程就会成为主STA,而工作线程随时可能中止,这种情况下,一旦工作线程中止主STA也就不复存在了,因此你需要在主线程中调用CoInitialize初始化主STA,即使主线程不使用任何COM组件。

MTA

多线程套间,所有COM组件都运行在本线程的MTA套件中。

这是就会出现多个线程同时执行某个COM调用,COM组件的开发者必须预料并处理这种并发访问带来的内存竞争读写混乱。COM组件开发者通常会应用临界区、互斥量、信号灯之类的常规线程同步方法。而调用者,不需要担心COM组件是否会因为多线程挂掉。

 如果一个COM组件是STA的,被错误的运行与MTA套件会引发各种奇怪的错误。

把STA的COM组件运行于MTA套件中会引发错误。

 

这就会引出一个问题,到底我该使用STA还是MTA呢,答案很简单问该COM组件的开发者,或者看他的说明文档,他们会告诉你。

 

STA套件的初始化方式(两种方式等效):

1,CoInitialize(nil);

2,CoInitializeEx(nil,COINIT_APARTMENTTHREADED);

 

MTA套间的初始化方式:

1,CoInitializeEx(nil,COINIT_MULTITHREADED);

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用C++调用COM组件需要遵循以下步骤: 1. 在Visual Studio中创建一个Windows控制台应用程序项目。 2. 在项目中添加一个头文件,该头文件包含COM组件的接口定义。 3. 在代码中使用CoCreateInstance函数创建COM组件的实例。 4. 调用COM组件的方法。 下面是一个简单的示例,展示如何使用C++调用COM组件的方法: ```cpp #include <windows.h> #include <tchar.h> #include <iostream> #include <comdef.h> // 定义COM组件的接口 struct IMyComponent : public IUnknown { virtual HRESULT __stdcall MyMethod() = 0; }; int main() { // 初始化COM库 CoInitialize(NULL); // 创建COM组件的实例 IMyComponent* pMyComponent = NULL; HRESULT hr = CoCreateInstance( CLSID_MyComponent, // COM组件的CLSID NULL, CLSCTX_INPROC_SERVER, // COM组件的类型 IID_IMyComponent, // COM组件的接口ID reinterpret_cast<void**>(&pMyComponent)); if (FAILED(hr)) { std::cout << "Failed to create instance of COM component." << std::endl; return 1; } // 调用COM组件的方法 hr = pMyComponent->MyMethod(); if (FAILED(hr)) { std::wcout << _com_error(hr).ErrorMessage() << std::endl; return 1; } // 释放COM组件的实例 pMyComponent->Release(); // 终止COM库 CoUninitialize(); return 0; } ``` 在上面的示例中,我们首先使用CoCreateInstance函数创建COM组件的实例,然后调用COM组件的MyMethod方法。 需要注意的是,如果COM组件是以Out-of-Process的方式运行的,我们需要使用CLSCTX_LOCAL_SERVER参数来创建COM组件的实例。另外,如果COM组件不是以STA(Single-Threaded Apartment)方式运行的,我们需要在调用CoInitialize之前设置线程模型为STA。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值