COM多线程原理与应用-程序员宅基地

http://blog.csdn.net/sheismylife/article/details/217033

目录:

COM多线程原理与应用... 1

目录:... 1

前言:... 1

套间:... 1

套间的定义:... 1

套间的分类:... 2

套间的进入和退出:.. 2

对象的同步:... 2

组件对象的同步:... 2

COM对象线程模型:.. 2

进程内对象线程模型的种类:.. 3

ATL对多线程的支持:.. 3

对象引用的保护:... 3

成员变量的保护:... 4

COM+导致的变化:... 4

上下文概述:... 4

上下文对象:... 4

调用对象:.. 5

 

前言:

COM多线程一直是个不容易弄清的问题,我也被困扰了很久,特别是COM在线程方面的术语总是不能统一。本文是为了将我所学所用得做一个总结,本文不保证一定正确,但是会随着时间的推移逐渐完善改正。

 

套间:

套间的定义:

       我个人认为<<COM技术内幕>>中关于套间的定义是错误的,应采用<<COM本质论>>中的定义。

<<COM技术内幕>>中-----

套间(Apartment),一个由用户界面线程(套间线程)和一个消息循环构成的概念性实体。

<<COM本质论>>中------

套间定义了一组对象的逻辑组合,这些对象共享同一组并发性和重入限制。一个线程要想使用COM,必须先进入一个套间。COM规定,只有运行在对象套间中的线程才能访问该对象。

套间的分类:

COM定义了两种类型的套间,STA(单线程套间)和MTA(多线程套间)。

STA的特点是套间内永远只有一个线程运行,并且一定是创建该套间的初始线程。因而开发只运行在STA中的组件不需要考虑线程同步等问题。

STA中包含了一个不可见的窗口,所以透过窗口的消息循环机制对消息队列的处理,保证了同一时刻只有一个调用请求被执行,这就是组件不需要处理同步的原因。

 

MTA的特点是套间内可以有多个线程运行,并且还可以创建新的线程。在MTA中运行的组件必须自己实现线程的同步。

 

 

套间的进入和退出:

线程通过CoInitializeEx函数进入套间,该函数的第二个参数通过传递COINIT_APARTMENTTHREADED和COINIT_MULTITHREADED标志了套间类型。如果传递COINIT_APARTMENTTHREADED,该线程将创建自己私有的套间,别人不能进入。如果传递COINIT_MULTITHREADED,该线程将进入当前进程范围内的MTA。

线程调用CoUninitialize函数用于退出所在的套间。只有退出后才可以进入其它类型的套间。

 

对象的同步:

组件对象的同步:

根据是否支持多线程同步,组件对象可以分为单线程对象和多线程对象。

COM对象线程模型:

每个COM对象都可以决定自己将运行在什么样的套间内,这称为COM对象的线程模型。

组件将运行在什么样的套间里面,这是由组件开发者决定的,调用组件的客户程序不需要关心。但是如果调用线程所在的套间和组件运行的套间不是同一套间,COM将插入代理/存根机制,调用线程在自身的套间内调用代理上的方法使用对象方法,而对象将运行在它需要的套间内。

进程内对象线程模型的种类:

线程模型的种类取决于注册表中ThreadingModel的值----------

1)  Both 表示组件可以运行在STA和MTA中

2)  Free 表示组件只能运行在MTA中

3)  Apartment表示组件只能运行在STA中

4)  为空表示组件只能运行在进程第一个创建的STA(主STA中)中。

如果调用线程的套间能够满足进程内组件的线程模型的要求,则直接创建组件对象,并不需要代理。

如果调用线程的套间不能满足进程内组件的线程模型的要求,则COM会在另一个合适的套间内(如果没有合适的套间,COM会创建一个)创建对象,然后给调用线程返回代理。

如果组件和客户程序不在同一个进程或者不在同一台机器,那必然是在两个套间中,COM会创建代理/存根。

如果组件对象的线程模型是Both或者Free,组件必须是多线程对象。

如果组件对象的线程模型是空,则组件可以是单线程对象或者多线程对象。这时候多线程对象内部的同步机制其实没有意义,因为不会有多个线程同时访问对象。

如果组件对象的线程模型是Apartment,情况将分两种:

a)       组件内部没有全局变量和静态变量,组件可以是单线程对象

b)      组件内部有全局变量和静态变量,组件应该是多线程对象,并对全局变量和静态变量进

行访问保护。

 

ATL对多线程的支持:

ATL中对COM对象中同步的支持有自己的考虑:如果COM对象本身就是不考虑同步的单线程对象,ATL就不应该对该对象的任何数据成员包括对象引用变量m_cRef进行同步保护。因为同步保护是需要额外的耗费资源的。ATL中的原则是只在应该需要保护的场合进行同步保护。这样做是符合ATL的设计目的----一切为了效率。

对象引用的保护:

CComSingleThreadModel、CComMultiThreadModel、CComMultiThreadModelNoCS类里面都定义了静态成员函数Increment和Decrement。CComObjectRootEx类的模板参数接受以上三个类并使用他们的静态成员函数,而我们的组件实现类从CComObjectRootEx类派生,从而获得了根据我们组件是否支持多线程而对对象引用进行同步的功能。一句话,我们只要派生具有合适的模板参数的CComObjectRootEx类就可以了。如下面我们的类:

class ATL_NO_VTABLE CMIS :

     public CComObjectRootEx<CComSingleThreadModel>,

CMIS类不需要考虑对象引用的同步保护,因为我们的组件对象是单线程对象,线程模型为空或者为Apartment。

       但是请注意ATL工程的组件类脚本中默认线程模型是both,这就会带来问题,所以我们手动修改它。

       如果我们改动代码如下:

class ATL_NO_VTABLE CMIS :

     public CComObjectRootEx< CComMultiThreadModel >,那么我们的组件对象引用计数就受到同步保护,而且我们的组件对象自己负责同步保护。这样如果CMIS类里有成员变量,那么我们要对成员变量进行同步保护。

成员变量的保护:

       采用临界区方式进行同步保护我们要利用类CComMultiThreadModel。我们的组件实现类派生自CComObjectRootEx< CComMultiThreadModel >类。使用的方法极其简单,我们只要在需要防止多个线程同时执行的函数内部开始处加上如下代码:

       ObjectLock Lock(this);

     如果想使用多个临界区的话使用方法就没这么简单了,但是除非必须,否则我不建议这样做,因为很容易引起死锁。一定要用的话必须先阅读<<Windows核心编程>>和<<Win32多线程程序设计>>这两本书。

 

COM+导致的变化:

上下文概述:

COM+对于套间的概念进一步的扩展,要求每个对象都必须运行在上下文中。通过上下文来控制对象的线程同步。

套间被细分为一个或者若干个上下文。一个上下文只能在一个套间中。一个套间至少包含一个默认上下文。

位于同一个套间中的不同上下文之间的对象调用必须经过轻量级代理的列集。轻量级代理不同于套间之间的代理,因为性能损失较小。

 

一个没有被COM+托管的传统COM组件不使用轻量级代理。但是该COM组件将被认为是放到了它所生存的套间的默认上下文内部,该套件的其他上下文(如果有的话)访问默认上下文不需要轻量级代理,默认上下文的引入只是为了和COM+上下文的概念一致。

 

上下文对象:

       上下文对象具体代表了每个上下文。CoGetObjectContext用来获取当前对象所在的上下文对象。

       上下文对象暴露了IObjectContextInfo、IContextState、IObjectContext、IObjectContextActivity四个接口。后面两个接口是向后兼容MTS,可以忽略。

       IObjectContextInfo::GetContextId方法可以获得上下文ID,调试时使用很方便。但是注意,只有COM组件被COM+管理后才能获得上下文对象,否则CoGetObjectContext将返回E_NOINTERFACE并返回NULL接口指针。所以如果我们开发的组件要同时适用于COM+和传统COM两种情况,就必须对CoGetObjectContext返回值进行查询并且作区分处理。

 

调用对象:

       当客户通过COM+服务调用COM对象时,COM+将创建名为“调用对象”的临时对象代表客户对COM对象的调用。调用对象将在方法返回后被销毁。

       调用对象暴露了两个与安全性设置相关的接口。

       CoGetCallContext函数可以让对象获取自己的调用对象,如果返回RPC_E_CALL_COMPLETE,则说明目前没有客户调用自己。

转载于:https://www.cnblogs.com/spinsoft/archive/2012/06/25/2561090.html

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_30784945/article/details/98574604

智能推荐

对每个边缘求最小外接矩形,通过最小矩形提取每个边缘_边缘的最小外接矩形-程序员宅基地

文章浏览阅读4.9k次。#include #include using namespace std;using namespace cv;int main(){Mat src; //源图像Mat tmp; //临时图像Mat dst_bw; //去掉背景后的目标二值图像Mat dst_contours;//轮廓图像src=imread("E:\\单板图片\\求孔洞数_边缘的最小外接矩形

【设计模式】中介者-程序员宅基地

文章浏览阅读865次。中介者,说白了跟市面上黑中介类似。当然这个中介,开发者是可以控制其行为的。也是在一定的信任关系上建立的。该模式要解决的问题是,一堆对象之间交叉耦合问题。网上看过群聊的例子。如果没有任何一个平台,多人之间的会话会是什么样的呢?不举多人,就三个吧A想把一句话说给BC,那么他首先要知道B和C在哪儿,然后分别告诉对方,自己想说的事情。如果再加一个人呢?问题很明显,此时各种群聊工具应运而生。我写

Mysql列自增是怎么实现的_mysql 自增序列生成原理-程序员宅基地

文章浏览阅读1.8k次。AUTO_INCREMENT两种情况1、在载入语句执行前,已经不确定要插入多少条记录。在执行插入语句时在表级别加一个auto-inc锁,然后为每条待插入记录的auto-increment修饰的列分配递增的值,语句执行结束后,再把auto-inc锁释放掉。一个事务再持有auto-inc锁的过程中,其他事务的插入语句都要被阻塞,可以保证一个语句中分配的递增值是连续的。AUTO-INC锁的..._mysql 自增序列生成原理

半导体器件基础_掺杂半导体的带隙-程序员宅基地

文章浏览阅读3.5k次,点赞2次,收藏17次。半导体能带结构示意图:上方两条白色带为没有电子填充的带,下面三条灰色带为充满电子的带,其中最高一条灰色带为价带,它与最低一条白色带之间的空隙为能隙空穴又称电洞(Electron hole),在固体物理学中指共价键上流失一个电子,最后在共价键上留下空位的现象导带(英语:conduction band),又名传导带,是指半导体或是绝缘体材料中,一种电子所具有能量的范围。这个能量的范围高..._掺杂半导体的带隙

基于C++和OpenCV的中心线提取算法_图像中心线提取c++-程序员宅基地

文章浏览阅读3.5k次,点赞2次,收藏26次。基于C++和OpenCV的中心线提取算法加权平方灰度重心法介绍算法演示加权平方灰度重心法介绍详情见 https://blog.csdn.net/u010518385/article/details/101015604算法演示下面展示 函数-输入图像和阈值,输出点。void get_median_line(Mat& src, int thresh, vector<Point2d>& points){ if (src.empty()) return; // 一、_图像中心线提取c++

HTML/CSS 常用单词整理_css上下外边距单词-程序员宅基地

文章浏览阅读8.3k次,点赞27次,收藏93次。页面布局(layout)header 头部/页眉;index 首页/索引;logo 标志;nav/sub_nav 导航/子导航;banner 横幅广告;main/content 主体/内容;container/con 容器;wrapper/wrap 包裹(类似于container);menu 菜单;sub_menu/second_menu 子菜单/二级菜单;..._css上下外边距单词

随便推点

Tomcat源码学习--WebAppClassLoader类加载机制-程序员宅基地

文章浏览阅读1.4w次,点赞6次,收藏21次。上一篇博客JVM-类加载机制中我们已经对JVM的类加载机制有所了解, 这篇博客我们了解一下Tomcat的类加载机制。Tomcat的类加载器可以分为两部分,第一个是Tomcat自身所使用的类加载器,会加载jre的lib包及tomcat的lib包的类,遵循类加载的双亲委派机制;第二个是每个Web应用程序用的,每个web应用程序都有自己专用的WebappClassLoader,优先加载/web-inf_webappclassloader

Idea打包失败|Failed to execute goal org.springframework.boot:spring-boot-maven-plugin:2.3.4.RELEASE-程序员宅基地

文章浏览阅读4.4k次,点赞2次,收藏5次。解决idea打包失败问题。_failed to execute goal org.springframework.boot:spring-boot-maven-plugin:2.3

Servlet.service() for servlet [dispatcherServlet] in context with path []-程序员宅基地

文章浏览阅读1.2k次。请看:(436条消息) JdbcTemplate空指针异常_刘贵庆的博客-程序员宅基地https://blog.csdn.net/xysxlgq/article/details/124323258_servlet.service() for servlet [dispatcherservlet] in context with path [/min

超全的莫队算法一遍过-程序员宅基地

文章浏览阅读3.6k次,点赞17次,收藏63次。莫队的基本思想、回滚莫队、树上莫队、二次离线,一文搞定。什么?你还不会莫队?看完这篇文章你就会了_莫队

vmware 安装 win10_vmware 退出iso-程序员宅基地

文章浏览阅读3.3k次。1. 下载安装vm软件,地址:VM下载2. 下载win10 iso镜像文件 (1) 下载win10安装工具:win10安装工具下载 (2) 下载完成后,选择给其他电脑安装ios文件即可得到win10的iso镜像文件(考验网速的时刻到了)。 ps:试过各种百度出来的镜像文件,发现没有能用的,还是乖乖通过官方方式下载吧。3. 安装win10 ios镜像文件 参..._vmware 退出iso

PTA-求函数的值_7-2 求函数的值编写程序,求s(x)前10项的和,x从键盘输入。-程序员宅基地

文章浏览阅读1.7k次,点赞2次,收藏3次。求函数的值:已知 s(x) = x -x^3/(3x1!)+x^5/(5x2!)-x^7/(7x3!)+... 。编写程序,求s(x)前10项的和,x从键盘输入。_7-2 求函数的值编写程序,求s(x)前10项的和,x从键盘输入。