我使用VS2008生成了一个ATL COM对象,代码包含对所调用定义的引用_MERGE_PROXYSTUB(因为我在最初运行向导时选择了'Merge proxy/stub'选项.)
代理/存根有什么意义?如果我没有选择合并选项,那么我会得到一个单独的MyControlPS.DLL- 什么时候会被使用?
如果我删除了_MERGE_PROXYSTUB定义包围的所有代码,那么控件似乎注册并正常工作.调试版本甚至没有定义_MERGE_PROXYSTUB,它仍然可以正常工作.
那么,我可以不使用代理/存根吗?
Joh*_*bly 16
如果希望使用与COM对象不同的线程模型从应用程序调用COM对象,则需要代理/存根.
例如,我们有一个插件,由使用特定线程模型的应用程序加载(不记得哪个),但我们的COM对象是多线程单元(MTA) - 因此需要代理/存根来编组在进行函数调用时,对象之间的数据,同时仍遵守线程模型的规则.
如果这些规则被破坏,那么COM将抛出异常或返回失败的HRESULT,例如RPC_E_WRONG_THREAD
如果不检查合并代理/存根选项,则visual studio会为代理/存根生成一个单独的项目,该项目将构建到单独的dll中.如果需要,这会使部署更加困难,但如果您不受线程模型问题的影响,您基本上可以忽略它们.
因此,如果调用COM对象的应用程序使用与对象相同的线程模型,则可以不使用代理/存根
Larry Osterman 在他的博客上提供了对线程模型的可读介绍.
另外,如果您的接口仅包含类型库友好的类型(BSTR、VARIANT 等)并出现在 IDL 的库块中,您可以选择让它们“编组类型库”,这意味着系统提供的代理/存根使用类型库中的元数据。
当接口被放入库块中,并且 DllRegisterServer 被定制为注册类型库时(如果我没记错的话,将 TRUE 传递给 XxxModule::DllRegisterServer),如有必要,您的接口将由系统进行编组,如 John Sible 所描述的那样。
此时,代理/存根甚至没有被使用,因此_MERGE_PROXYSTUB没有任何效果。