12

我使用 VS2008 生成了一个 ATL COM 对象,并且代码包含对名为的定义的引用_MERGE_PROXYSTUB(因为我在最初运行向导时选择了“合并代理/存根”选项。)

代理/存根有什么意义?如果我不选择合并选项,那么我会得到一个单独的选项MyControlPS.DLL- 什么时候会使用它?

如果我删除定义包围的所有代码,FWIW 控件似乎可以注册并正常工作_MERGE_PROXYSTUB。调试版本甚至没有定义_MERGE_PROXYSTUB,它仍然可以正常工作。

那么,我可以不用代理/存根吗?

4

2 回答 2

17

如果您希望使用与 COM 对象不同的线程模型从应用程序调用您的 COM 对象,则需要代理/存根。

例如,我们有一个由使用特定线程模型(不记得是哪个)的应用程序加载的插件,但我们的 COM 对象是多线程单元 (MTA) - 所以需要代理/存根来编组进行函数调用时对象之间的数据,同时仍然遵守线程模型的规则。

如果这些规则被破坏,那么 COM 将抛出异常或返回失败 HRESULT,例如RPC_E_WRONG_THREAD

如果您不检查合并代理/存根选项,则 Visual Studio 会为代理/存根生成一个单独的项目,该项目将构建到一个单独的 dll 中。如果需要,这会使部署变得更加困难,但是如果您不受线程模型问题的影响,您基本上可以忽略它们。

因此,如果调用 COM 对象的应用程序使用与您的对象相同的线程模型,您可以不使用代理/存根

Larry Osterman在他的博客上提供了关于线程模型的可读介绍。

于 2009-09-18T13:37:31.203 回答
4

此外,如果您的接口仅包含类型库友好类型(BSTR、VARIANT 等)并出现在 IDL 的库块中,您可以选择让它们“编组类型库”,这意味着系统提供的代理/存根使用类型库中的元数据。

当接口被放入库块中,并且 DllRegisterServer 被定制为注册类型库(如果我没记错的话,将 TRUE 传递给 XxxModule::DllRegisterServer)你的接口将由系统编组,如有必要,如 John Sibly 所述。

那时,甚至没有使用代理/存根,因此_MERGE_PROXYSTUB没有任何效果。

于 2009-09-21T04:59:19.893 回答