MSDN上没有CoGetClassObject函数的源码,资料上又说CoGetClassObject调用DllGetClassObject函数.不知这其中是如何调用地? //================= 查了一些资料 摘录1如下: ---------- CoCreateInstance API函数将调用CoGetClassObject API,这个调用过程我们是看不到相关的代码的,因为微软已经把他封装好了。而CoGetClassObject函数的作用是什么呢?它将调用 LoadLibrary来寻找我们指定的COM组件(DLL),然后使用GetProcAddress 来寻找组件的入口函数 - 其中的DllGetClassObject 函数就在这里将被调用。
摘录2如下: ---------- CoGetClassObject及CoCreateInstance中都有对LoadLibrary和 GetProcAddress进行调用。其中LoadLibrary的参数为DLL的名称,而CoGetClassObject及 CoCreateInstance则使用它们的第一个参数(一个CLSID),组件可以用CLSID作为索引在Windows注册表中发布包含它们的 DLL文件名称。CoGetClassObject及CoCreateInstance将用CLSID作为关键字在注册表中查询所需的DLL文件名称。 //================ 当CoGetClassObject调用DllGetClassObject后,发现指针与句柄的关系就是(句柄是指针的指针) 从这俩个函数的参数就可以看出 STDAPI CoGetClassObject( REFCLSID rclsid, DWORD dwClsContext, COSERVERINFO * pServerInfo, REFIID riid, LPVOID * ppv ); STDAPI DllGetClassObject( const CLSID & rclsid, const IID & riid, void ** ppv ); 具体函数例子: IClassFactory *classFactory; CoGetClassObject(&CLSID_IExample, CLSCTX_INPROC_SERVER, 0, &IID_IClassFactory, &classFactory)
HRESULT PASCAL DllGetClassObject(REFCLSID objGuid, REFIID factoryGuid, void **factoryHandle) { register HRESULT hr; if (IsEqualCLSID(objGuid, &CLSID_IExample)) { hr = classQueryInterface(&MyIClassFactoryObj, factoryGuid, factoryHandle); } else { *factoryHandle = 0; hr = CLASS_E_CLASSNOTAVAILABLE; } return(hr); }
CoGetClassObject中的参数&classFactory 与DllGetClassObject中的参数void **factoryHandle //===================== CoGetClassObject要加载DLL,但是我们并没有给他传递DLL的路径(DLL相关请参考DLL-使用DLL)。所以它要用参数clsid(即组件id,它是从CoCreateInstance传递过来的)去注册表找到这个组件DLL路径(请见:COM笔记-Widows注册表),然后把DLL加载起来。 加载起来之后我们就可以调用DLL的导出函数DllGetClassObject。并把clsid(组件ID),IID_IclassFactory(接口ID)传递它。(可以看一下DllGetClassObject的实现,看看COM笔记-类厂的源代码)。 DllGetClassObject执行成功则返回类工厂接口pIFactory(见图)。 接着就是 类工厂接口pIFactory 调用 它的方法CreateInstance,去创建我们需要的组件,并返回我们需要的接口了。(同样可以看看COM笔记-类厂的源代码) 关于参数punkonwnDuter这个没用传递CoGetClassObject,而是传递给了类厂的一个方法CreateInstance,主要是聚合方面的东西。(见:COM笔记-包容与聚合) 参数dwClsContext限定所创建的组件的执行上下文。(见:COM笔记-CoCreateInstance)
|