Atl笔记二:BEGIN_COM_MAP

1,offsetofclass
获取基类相对于子类的偏移位置。

#define _ATL_PACKING 8
#define
offsetofclass(base, derived) ((DWORD_PTR)(static_cast<base*>((derived*)_ATL_PACKING))-_ATL_PACKING)

_ATL_PACKING非零就行,只是作为一个地址。因为为了避免虚类无法创建对象的问题所以没有通过类对象来计算。

2,
//If you get a message that FinalConstruct is ambiguous then you need to
// override it in your class and call each base class‘ version of this
#define
BEGIN_COM_MAP(x) public: \
????typedef
x
_ComMapClass; \
????static
HRESULT
WINAPI
_Cache(void* pv, REFIID
iid, void** ppvObject, DWORD_PTR
dw) throw()\
????{\
????????_ComMapClass* p = (_ComMapClass*)pv;\
????????p->Lock();\
????????HRESULT
hRes = E_FAIL; \
????????__try \
????????{ \
????????????hRes = ATL::CComObjectRootBase::_Cache(pv, iid, ppvObject, dw);\
????????} \
????????__finally \
????????{ \
????????????p->Unlock();\
????????} \
????????return
hRes;\
????}\
????IUnknown* _GetRawUnknown() throw() \
????{ ATLASSERT(_GetEntries()[0].pFunc == _ATL_SIMPLEMAPENTRY); return (IUnknown*)((INT_PTR)this+_GetEntries()->dw); }
\
????_ATL_DECLARE_GET_UNKNOWN(x)\
????HRESULT
_InternalQueryInterface(REFIID
iid, void** ppvObject) throw() \
????{ return
InternalQueryInterface(this, _GetEntries(), iid, ppvObject); } \
????const
static
ATL::_ATL_INTMAP_ENTRY* WINAPI
_GetEntries() throw() { \
????static
const
ATL::_ATL_INTMAP_ENTRY
_entries[]
= { DEBUG_QI_ENTRY(x)

struct
_ATL_INTMAP_ENTRY
{
????const
IID* piid; // the interface id (IID)
????DWORD_PTR
dw;
????_ATL_CREATORARGFUNC* pFunc; //NULL:end, 1:offset, n:ptr
};

#define
DEBUG_QI_ENTRY(x) \
????????{NULL, \
????????(DWORD_PTR)_T(#x), \
????????(ATL::_ATL_CREATORARGFUNC*)0},

typedef
HRESULT (WINAPI
_ATL_CREATORARGFUNC)(void* pv, REFIID
riid, LPVOID* ppv, DWORD_PTR
dw);

#define
COM_INTERFACE_ENTRY(x)\
????{&_ATL_IIDOF(x), \
????offsetofclass(x, _ComMapClass), \
????_ATL_SIMPLEMAPENTRY},

#define
_ATL_IIDOF(x) __uuidof(x)
//__uuidof获取与x相关的GUID值

#define
_ATL_SIMPLEMAPENTRY ((ATL::_ATL_CREATORARGFUNC*)1)

#define
END_COM_MAP() \
????__if_exists(_GetAttrEntries) {{NULL, (DWORD_PTR)_GetAttrEntries, _ChainAttr }, }\
????{NULL, 0, 0}}; return &_entries[1];} \
????virtual
ULONG
STDMETHODCALLTYPE
AddRef( void) throw() = 0; \
????virtual
ULONG
STDMETHODCALLTYPE
Release( void) throw() = 0; \
????STDMETHOD(QueryInterface)(REFIID, void**) throw() = 0;

3,
static
HRESULT
WINAPI CComObjectRootBase ::InternalQueryInterface(void* pThis,
????????const
_ATL_INTMAP_ENTRY* pEntries, REFIID
iid, void** ppvObject)
????{
????????// Only Assert here. AtlInternalQueryInterface will return the correct HRESULT if ppvObject == NULL
#ifndef
_ATL_OLEDB_CONFORMANCE_TESTS
????????ATLASSERT(ppvObject != NULL);
#endif
????????ATLASSERT(pThis != NULL);
????????// First entry in the com map should be a simple map entry
????????ATLASSERT(pEntries->pFunc == _ATL_SIMPLEMAPENTRY);
????#if
defined(_ATL_DEBUG_INTERFACES) || defined(_ATL_DEBUG_QI)
????????LPCTSTR pszClassName = (LPCTSTR) pEntries[-1].dw;
????#endif
// _ATL_DEBUG_INTERFACES
????????HRESULT
hRes = AtlInternalQueryInterface(pThis, pEntries, iid, ppvObject);
????#ifdef
_ATL_DEBUG_INTERFACES
????????_AtlDebugInterfacesModule.AddThunk((IUnknown**)ppvObject, pszClassName, iid);
????#endif
// _ATL_DEBUG_INTERFACES
????????return
_ATLDUMPIID(iid, pszClassName, hRes);
????}

4,
CComObjectRootBase中以表驱动的方式对接口的查询作了一个内部的实现,即InternalQueryInterface()。
所以在创建基于ATL的COM类时,需要创建一个包含所有实现接口的映射表。
ATL提供了BEGIN_COM_MAP、END_COM_MAP、COM_INTERFACE_ENTRY与COM_INTERFACE_ENTRY2这4个宏来创建接口映射表。
假设一个类CClassA继承了接口IIntA1和IIntA2,则该类的接口映射表创建如下:

class CClassA : public CComObjectRootEx<CComSingleThreadMode>
{
??? BEGIN_COM_MAP(CClassA)
??????? COM_INTERFACE_ENTRY(IIntA1)
??????? COM_INTERFACE_ENTRY(IIntA2)
??? END_COM_MAP()
??? ......
};

static
const
ATL::_ATL_INTMAP_ENTRY
_entries[] =
{
{NULL, ????(DWORD_PTR)_T(#x), ????(ATL::_ATL_CREATORARGFUNC*)0},
{& __uuidof(IIntA1), offsetofclass(IIntA1,
CClassA), (ATL::_ATL_CREATORARGFUNC*)1},
{& __uuidof(IIntA2), offsetofclass(IIntA2,
CClassA), (ATL::_ATL_CREATORARGFUNC*)1},
{NULL, 0, 0}
};

而当CClassB继承了IIntB1和IIntB2,并且IIntB1和IIntB2都继承自IDispatch接口。
此时,如果客户程序在查询IDispatch接口,QueryInterface所返回的IDispatch接口指针将无法确定其属于IIntB1还是IIntB2。
在这种情况下,需要指定IDispatch接口指针的默认指向。?COM_INTERFACE_ENTRY2()宏即是用于完成该功能。
下面代码将对IDispatch接口的请求默认指向属于IIntB2的IDispatch接口指针。
class CClassB : public CComObjectRootEx<CComSingleThreadMode>
{
??? BEGIN_COM_MAP(CClassB)
??????? COM_INTERFACE_ENTRY(IIntB1)
??????? COM_INTERFACE_ENTRY(IIntB2)
??????? COM_INTERFACE_ENTRY2(IDispatch, IIntB2)
??? END_COM_MAP()
??? ......
};

#define COM_INTERFACE_ENTRY2(x, x2)\
????{&_ATL_IIDOF(x),\
????reinterpret_cast<DWORD_PTR>(static_cast<x*>(static_cast<x2*>(reinterpret_cast<_ComMapClass*>(8))))-8,\
????_ATL_SIMPLEMAPENTRY},

5,总结:
BEGIN_COM_MAP通过一个静态的_GetEntries()方法,来获取在该方法中创建的一个静态COM接口映射表。

时间: 2024-12-28 17:43:04

Atl笔记二:BEGIN_COM_MAP的相关文章

老男孩培训视频听课笔记二(在51cto上听的)

centos 5.8 文本安装过程    引导采用默认,引导不用设置密码    网络配置,根据实际情况配置,网关是网络出口的地址,一般为wlan出口的路由器的地址或者是代理服务器的内网IP    DNS简单解说图:      主机名--时区--root密码    选择自定义系统安装包--最小化(安全方便工作,建议安装以下的组)      ·base-- 基础      ·editors-编辑器      ·development librarays--开发库      ·development

《卓有成效的程序员》----读书笔记二

六大方面对比Launchy和TypeAndRun(TAR) 对于快速启动工具,很多人都有自己的偏好,多次听到朋友介绍Launchy的好,虽然自己一直在使用着TAR,还是克制不住对于好软件的渴求,下载Launchy进行试用.很多软件都是有一个试用期的,也许新的软件确实不错,但是你习惯了以前使用的那个软件.今天就比较客观的将Launchy和TAR进行一下对比,从界面.上手速度到功能.自定义,以及软件的稳定性.占用资源进行详细的比较. [界面美观]Launchy:毫无疑问这是它的强项.1.0正式版自带

Caliburn.Micro学习笔记(二)----Actions

Caliburn.Micro学习笔记(二)----Actions 上一篇已经简单说了一下引导类和简单的控件绑定 我的上一个例子里的button自动匹配到ViewModel事件你一定感觉很好玩吧 今天说一下它的Actions,看一下Caliburn.Micro给我们提供了多强大的支持 我们还是从做例子开始 demo的源码下载在文章的最后 例子1.无参数方法调用 点击button把textBox输入的文本弹出来 如果textbox里没有文本button不可点,看一下效果图 看一下前台代码 <Stac

《逻辑思维简易入门》(第2版) 阅读笔记二

<逻辑思维简易入门>(第2版) 阅读笔记二 本周阅读的是<逻辑思维简易入门>的第三章,也就是说,本书的第一部分就已经读完了. 第三章.信念的优点 信念和负信念是人们在接受一个事物时一种心理态度,延伸来说也就是对事物的认知态度.因为我们在研究 逻辑思维的时候,都有一个前提:“以正常情况以及说话者真诚”,所以有人如果对于一件事物不做回应,我们可以认为这是一种既不相信,也不怀疑的的态度. 信念的优缺点有很多,在书中主要介绍了下面几种: 1.准确性 好的信念实在准确的表达事实,同样真的信念

2. 蛤蟆Python脚本学习笔记二基本命令畅玩

2. 蛤蟆Python脚本学习笔记二基本命令畅玩 本篇名言:"成功源于发现细节,没有细节就没有机遇,留心细节意味着创造机遇.一件司空见惯的小事或许就可能是打开机遇宝库的钥匙!" 下班回家,咱先来看下一些常用的基本命令. 欢迎转载,转载请标明出处:http://blog.csdn.net/notbaron/article/details/48092873 1.  数字和表达式 看下图1一就能说明很多问题: 加法,整除,浮点除,取模,幂乘方等.是不是很直接也很粗暴. 关于上限,蛤蟆不太清楚

Emacs 笔记二

Emacs 笔记二 Table of Contents 1. 前言 2. emacs基本操作(常用快捷键) 3. emacs模式讲解 4. emacs缓冲区 5. org mode 5.1. 列表 5.2. 快键键 5.3. 内嵌元素(插入代码什么的) 5.4. 表格 1 前言 最近在学着写博客,发现MarkDown真乃神器,于是去找了很多markdown的工具,发现作业部落 最好的那个,而无意间又发现了org-mode火爆到极致 非常被人推崇,其实作业部落 已经是能很完美的满足我的需求了,但是

《Programming in Lua 3》读书笔记(二十二)

日期:2014.8.6 PartⅣ The C API 26 Extending Your Application 使用Lua很重要的一点是用来做配置语言.配合主语言做一些功能的配置. 26.1 The Basics 有的时候程序需要配置一些功能信息,很多时候可能有许多别的方法比用lua做配置要更简单:如使用环境变量或者读取文件,读取文件涉及到文件的解析.如果使用Lua进行配置的话,相当于用lua文件替代了要读取的如csv.txt文件等. 使用Lua进行配置的时候,就需要使用Lua API去控制

小猪的数据结构学习笔记(二)

小猪的数据结构学习笔记(二) 线性表中的顺序表 本节引言: 在上个章节中,我们对数据结构与算法的相关概念进行了了解,知道数据结构的 逻辑结构与物理结构的区别,算法的特性以及设计要求;还学了如何去衡量一个算法 的好坏,以及时间复杂度的计算!在本节中我们将接触第一个数据结构--线性表; 而线性表有两种表现形式,分别是顺序表和链表;学好这一章很重要,是学习后面的基石; 这一节我们会重点学习下顺序表,在这里给大家一个忠告,学编程切忌眼高手低,看懂不代表自己 写得出来,给出的实现代码,自己要理解思路,自己

JavaScript--基于对象的脚本语言学习笔记(二)

第二部分:DOM编程 1.文档象模型(DOM)提供了访问结构化文档的一种方式,很多语言自己的DOM解析器. DOM解析器就是完成结构化文档和DOM树之间的转换关系. DOM解析器解析结构化文档:将磁盘上的结构化文档转换成内存中的DOM树 从DOM树输出结构化文档:将内存中的DOM树转换成磁盘上的结构化文档 2.DOM模型扩展了HTML元素,为几乎所有的HTML元素都新增了innerHTML属性,该属性代表该元素的"内容",即返回的某个元素的开始标签.结束标签之间的字符串内容(不包含其它