CoCreateInstance具体内部实现

转载 2013年12月06日 10:25:31
  1.       CoCreateInstance(....)   
  2.     {   
  3.     //.......    
  4.     IClassFactory *pClassFactory=NULL;   
  5.     CoGetClassObject(CLSID_Object, CLSCTX_INPROC_SERVER, NULL, IID_IClassFactory, (void **)&pClassFactory);   
  6.     pClassFactory->CreateInstance(NULL, IID_IUnknown, (void**)&pUnk);   
  7.     pClassFactory->Release();   
  8.     //........    
  9.    }    

  这段话的意思就是先得到类厂对象,再通过类厂创建组件从而得到IUnknown指针。继续深入一步,看看CoGetClassObject的内部伪码: 

   
  1.     CoGetClassObject(.....)   
  2.    {   
  3.     //通过查注册表CLSID_Object,得知组件DLL的位置、文件名    
  4.     //装入DLL库    
  5.     //使用函数GetProcAddress(...)得到DLL库中函数DllGetClassObject的函数指针。    
  6.     //调用DllGetClassObject    
  7.    }   
  8.    /// DllGetClassObject是干什么的,它是用来获得类厂对象的。只有先得到类厂才能去创建组件.    
  9.    /// 下面是DllGetClassObject的伪码:    
  10.     DllGetClassObject(...)   
  11.     {   
  12.     //......    
  13.     CFactory* pFactory= new CFactory; //类厂对象    
  14.     pFactory->QueryInterface(IID_IClassFactory, (void**)&pClassFactory);   
  15.     //查询IClassFactory指针    
  16.     pFactory->Release();   
  17.     //......    
  18.     }   
  19.    /// CoGetClassObject的流程已经到此为止,现在返回CoCreateInstance,看看CreateInstance的伪码:    
  20.     CFactory::CreateInstance(.....)   
  21.     {   
  22.     //...........    
  23.     CObject *pObject = new CObject; //组件对象    
  24.     pObject->QueryInterface(IID_IUnknown, (void**)&pUnk);   
  25.     pObject->Release();   
  26.     //...........    
  27.     }   

这部分我们将构造一个创建COM组件的最小框架结构,然后看一看其内部处理流程是怎样的

  1.     IUnknown *pUnk=NULL;  
  2.     IObject *pObject=NULL;  
  3.     CoInitialize(NULL);  
  4.     CoCreateInstance(CLSID_Object, CLSCTX_INPROC_SERVER, NULL, IID_IUnknown, (void**)&pUnk);  
  5.     pUnk->QueryInterface(IID_IOjbect, (void**)&pObject);  
  6.     pUnk->Release();  
  7.     pObject->Func();  
  8.     pObject->Release();  
  9.     CoUninitialize();  


  1. CoCreateInstance(....)  
  2.  {  
  3.     .......  
  4.     IClassFactory *pClassFactory=NULL;  
  5.     CoGetClassObject(CLSID_Object, CLSCTX_INPROC_SERVER, NULL, IID_IClassFactory, (void **)&pClassFactory);  
  6.     pClassFactory->CreateInstance(NULL, IID_IUnknown, (void**)&pUnk);  
  7.     pClassFactory->Release();  
  8.     ........  
  9.  }  

  这就是一个典型的创建COM组件的框架,不过我的兴趣在CoCreateInstance身上,让我们来看看它内部做了一些什么事情。以下是它内部实现的一个伪代码:

  1.   CoGetClassObject(.....)  
  2.    {  
  3.     //通过查注册表CLSID_Object,得知组件DLL的位置、文件名   
  4.     //装入DLL库   
  5.     //使用函数GetProcAddress(...)得到DLL库中函数DllGetClassObject的函数指针。   
  6.     //调用DllGetClassObject   
  7.    }  
  8.     DllGetClassObject是干什么的,它是用来获得类厂对象的。只有先得到类厂才能去创建组件.  
  9.     下面是DllGetClassObject的伪码:  
  10.     DllGetClassObject(...)  
  11.     {  
  12.     ......  
  13.     CFactory* pFactory= new CFactory; //类厂对象   
  14.     pFactory->QueryInterface(IID_IClassFactory, (void**)&pClassFactory);  
  15.     //查询IClassFactory指针   
  16.     pFactory->Release();  
  17.     ......  
  18.     }  
  19.     CoGetClassObject的流程已经到此为止,现在返回CoCreateInstance,看看CreateInstance的伪码:  
  20.     CFactory::CreateInstance(.....)  
  21.     {  
  22.     ...........  
  23.     CObject *pObject = new CObject; //组件对象   
  24.     pObject->QueryInterface(IID_IUnknown, (void**)&pUnk);  
  25.     pObject->Release();  
  26.     ...........  
  27.     }  

  这段话的意思就是先得到类厂对象,再通过类厂创建组件从而得到IUnknown指针。继续深入一步,看看CoGetClassObject的内部伪码:


  

上图是从COM+技术内幕中COPY来的一个例图,从图中可以清楚的看到CoCreateInstance的整个流程。

  (7) 一个典型的自注册的COM DLL所必有的四个函数

  DllGetClassObject:用于获得类厂指针
  DllRegisterServer:注册一些必要的信息到注册表中
  DllUnregisterServer:卸载注册信息
  DllCanUnloadNow:系统空闲时会调用这个函数,以确定是否可以卸载DLL

  DLL还有一个函数是DllMain,这个函数在COM中并不要求一定要实现它,但是在VC生成的组件中自动都包含了它,它的作用主要是得到一个全局的实例对象。

  (8) 注册表在COM中的重要作用

  首先要知道GUID的概念,COM中所有的类、接口、类型库都用GUID来唯一标识,GUID是一个128位的字串,根据特制算法生成的GUID可以保证是全世界唯一的。 COM组件的创建,查询接口都是通过注册表进行的。有了注册表,应用程序就不需要知道组件的DLL文件名、位置,只需要根据CLSID查就可以了。当版本升级的时侯,只要改一下注册表信息就可以神不知鬼不觉的转到新版本的DLL。

COM学习笔记(二)CoCreateInstance具体内部实现

2012-11-22 09:11 1554人阅读 评论(0) 收藏 举报 comCOMwindowsWindowsWINDOWSWIndows [cpp] view pla...
  • Andeewu
  • Andeewu
  • 2013年03月31日 14:47
  • 1009

关于COM对象创建(CoCreateInstance,与QueryInterface)

panda2002-12-19 05:40 PM一个土问题,关于COM对象的创建 因为生活所迫,这几天必须对COM有所了解。今天看了之后,对COM对象的创建有所困惑。使用CoCreateInstanc...
  • flyfwater
  • flyfwater
  • 2008年05月31日 10:57
  • 2383

使用CoGetClassObject而不使用CoCreateInstance的情况

使用CoGetClassObject而不使用CoCreateInstance的情况  大多数情况下,组件的创建均使用CoCreateInstance而不是用CoGetClassObject,但是在如下...
  • dj0379
  • dj0379
  • 2009年12月26日 00:49
  • 3605

CoCreateInstance 使用及参数说明

CoCreateInstance 创建组件的最简单的方法是使用CoCreateInstance函数。 在COM库中包含一个用于创建组件的名为CoCreateInstance的函数。此函数需要一个...
  • yum2006
  • yum2006
  • 2011年12月20日 11:44
  • 1272

CoCreateInstance 出错 ,返回 -2147221164 CLASS_NOT_REGISTERED ,原因

今天发现有台机子执行 CoCreateInstance 总是不成功,而在其他的很多机子上是没有问题的。追了半天发现这台机子的注册表权限不对。HKEY_CLASSES_ROOT的权限只有Everyone...
  • huigll
  • huigll
  • 2006年12月25日 11:29
  • 4701

CoCreateInstance调用COM接口伪流程

在编写组件程序时,经常会使用CoCreateInstance直接取COM组件的接口,非常方便,那CoCreateInstance到底干了些什么事呢?1、CoCreateInstance取COM组件的接...
  • gufeng99
  • gufeng99
  • 2015年05月15日 23:49
  • 938

QT5.9 在线程中导出EXCLE报 CoCreateInstance failure (尚未调用 CoInitialize) 解决办法

错误截图: 原因:因为COM是在GUI线程里初始化和销毁的。在新开的线程里并没有初始化,所以得自己初始化 解决办法: 1.引用头文件  #include "qt_windows.h" ...
  • woailyoo0000
  • woailyoo0000
  • 2017年08月29日 23:11
  • 193

调用CoCreateInstance 函数 返回值-2147221008 错误

调用之前 加HRESULT Hr = ::CoInitialize(NULL); 调用之后加::CoUninitialize(); 注意CoInitialize 和 CoUninitialize ...
  • mail_cm
  • mail_cm
  • 2012年05月16日 09:41
  • 1179

CoCreateInstance of OLE control failed

1.在将VC6.0的工程转换到VS2005下编译连接都没有问题运行时报错Assert fail in occcont.cpp at line 950 用Windbg调试可执行程序发现有如下错误CoCr...
  • cwcww1314
  • cwcww1314
  • 2015年09月26日 16:33
  • 932

CoCreateInstance失败,返回错误码0x8007007e

CoCreateInstance失败,返回错误码0x8007007eDebug编译的COM接口好好的,但用Release编译后,用CoCreateInstance获得接口都失败。         Re...
  • dj0379
  • dj0379
  • 2009年12月21日 17:07
  • 4887
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:CoCreateInstance具体内部实现
举报原因:
原因补充:

(最多只允许输入30个字)