开始先说一句,学习com之前要学好c++ 对象模型。


QueryInterface的使用:

QueryInterface是IUnknown的一个成员函数,客户可以通过此函数来查询某个组件是否支持某个特定的接口。

QueryInterface成功返回一个指向此接口的指针。     错误返回一个错误代码。


HRESULT _stdcall QueryInterface(const IID& iid, void **ppv);



 

下面是一个QueryInterface的使用实例:(可知道相应的组件是否支持某个特定的接口)

 

void ceshi(IUnknown * pI)
{
IX* pIX = NULL;
HRESULT hr = pI->QueryInterface(IID_IX, (void**)&pIX);
if(SUCCEEDED(hr))
{
pIX->Fx();
}
}


我们查询pI 是否支持由IID_IX所标识的接口。

 

代码中主意的是   pIX需要初始化为NULL, 这是一种比较好的编程方法,在后面的实现中我们会看到  QueryInterface在失败时将把返回的接口指针置为NULL。

由于QueryInterface是由程序员而不是由系统实现的,因此某些组件可能并不会在查询失败时将此指针置为NULL。    为了安全,在程序中还是我们自己将其置为NULL比较好。


QueryInterface的实现:

   他的实现需要完成的不过是根据某个给定的IID返回指向相应接口的指针。

若组件支持客户指定的接口,那么应返回S_OK 以及相应的指针。

若不支持,返回值应为: E_NOINTERFACE 并将相应的指针返回值置为NULL。

 

HRESULT _stdcall CA::QueryInterface(const IID& iid, void ** ppv)
{
if(iid == IID_IUnknown)
{
*ppv = static_cast<IX*>(this);
}else if(iid == IID_IX)
{
*ppv = static_cast<IX*>(this);
}else if(iid == IID_IY)
{
*ppv = static_cast<IY*>(this)
}else
{
*ppv = NULL;
return E_NOINTERFACE;
}
static_cast<IUnknown*>(*ppv)->AddRed();
return S_OK;
}


QueryInterface 的末尾调用AddRef实际上没有任何作用,后面讨论 AddRef;

 


QueryInterface的实现规则:

1.QueryInterface返回的总是同一IUnknown指针。

2.若客户曾经获取过某个接口,那么它将总能获取此接口。

3.客户可以再次获取已经拥有的接口。

4.客户可以返回到起始接口。

5.若能从某个接口获取某特定接口,那么可以从任意接口都将可以获取此接口。



QueryInterface定义了组件:

      QueryInterface是COM最为重要的部分,因为一个组件实际上就是由QueryInterface定义了,组件所支持的接口集就是QueryInterface能够为之返回接口指针的那些接口。

这是由QueryInterface的实现决定的,而不是由实现组件的C++类决定的。

  客户不知道QueryInterface的实现,他将无法知道一个组件所支持的所有接口。   客户了解组件所支持接口的唯一方法是进行查询。

   COM类似于在某次社交聚会上同某人会面,而与对他们进行工作面试有很大不同,当进行工作面试时,被试者将提交一份介绍他情况的个人简历。这份简历类似于c++类的定义。而社交聚会上会面时,没有人会给对方提供个人简历。为了了解情况,必须询问。   这一点类似于COM组件。



新版本的处理:

Com中,接口是不会发生改变的,当组件发布一个接口并被某个客户使用之后,此接口将绝不会发生任何变化,而将永远保持不变。

每一个接口都有一个唯一的接口标识符IID,  一般情况下,我们不会改变接口,而可以建立一个新接口并为之指定一个新的IID。

当QueryInterface 接收到对老IID  的查询时,他将返回老接口,   而当他收到对新的IID的查询时,他将返回新的接口。   

对QueryInterface 而言, 一个IID 就是一个接口。


新的接口可以继承老接口,他也可以同老接口完全不同。  由于老接口仍然保持不变,已有客户的运行将不会收到任何影响。

新客户可以自行决定是使用老接口还是新接口,因为他可以自由决定到底查询那个接口。


接口的IID  据定了它的版本,当客户获取某个接口时,由于不同版本的接口实际上是不同的接口,他们各自具有不同的ID,因此客户仍能取得正确的版本的接口。


我们何时需要建立一个新版本?

当改变下列条件中的任何一个时,就应该给新街口指定新的id。

1、接口中函数的数目;

2、接口中函数的顺序;

3、某个函数的参数;

4.、某个函数参数的顺序;

5、某个函数参数的类型;

6、函数可能的返回值;

7、函数返回值的类型;

8、函数参数的含义;

9、接口中函数的含义。


   只要是所做的修改会导致已有客户的正常运行,就需要接口指定新的ID。




2013.8.4

jofranks 于南昌