首先来分析上述代码的类继承结构
IUnknown是原始抽象基类, Derived1和Derived2都是从IUnknown派生的(两个都是接口),A 从Derived1和Derived2继承的。
现在我们来看看三个类的对象模型布局:
从图可以看出各个接口的虚函数表, 值得注意的是类A由于实现了所有的接口,所以虚函数表会有所改变。
所以当QueryInterface得到接口Derived1或者Derived2接口指针时候调用AddRef 或者Release函数时候都是调用A实现里面的相应版本,这就是所谓的动态绑定技术,当然组件对象必须是A类型的啦。
还有当查询IID_IUnknown接口时,如果QueryInterface是这样实现的
if (g_strIUnknown == iid)
{
*ppvObject = (IUnknown*)this; // 因为是多继承,所以在组件对象中有两个Base对象,这种转换编译器会报错
}
编译器报错, 原因是:因为是多继承,所以在组件对象中有两个Base对象,造成二义性。这种转换编译器会报错
只要把 *ppvObject = (IUnknown*)this;改成*ppvObject = (Derived1*)this; 就行了。
当然本文只是用简单的C++ 来模型了一下COM 组件对象, 真正的COM 编程比这个要复杂点, 推荐大家去看看<<COM原理与应用>>