如何从 .NET 对 .NET 中定义的 COM 对象执行后期绑定调用(通过 DispID)?
Posted
技术标签:
【中文标题】如何从 .NET 对 .NET 中定义的 COM 对象执行后期绑定调用(通过 DispID)?【英文标题】:How to perform late-bound invocation (via DispID) of COM object defined in .NET from .NET? 【发布时间】:2012-02-14 20:31:47 【问题描述】:我想从另一个使用 DispID 的 .NET 应用程序调用在 .NET 中定义的 COM 对象的方法。
我在 .NET 项目中定义了以下类和接口:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
namespace TestComObject
[ComVisible(true)]
[InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
[Guid("99476c04-574a-4271-abc8-f1d2615c3d93")]
public interface IKnowEverything
[DispId(1030)]
string GetTheTruth();
[ComVisible(true)]
[Guid("9B869BAF-622A-458B-BF80-2CD6D8A3C541")]
[ClassInterface(ClassInterfaceType.None)]
public class AllKnowing : IKnowEverything
public string GetTheTruth()
return "I am King!";
我构建它并设置为注册 COM 互操作,它使用强密钥签名。
现在我想使用来自另一个 .NET 应用程序的 DispID 调用此方法。我写了如下测试方法:
static void Main(string[] args)
Guid clsid = new Guid("9B869BAF-622A-458B-BF80-2CD6D8A3C541");
Type type = Type.GetTypeFromCLSID(clsid, true);
object instance = Activator.CreateInstance(type);
string methodName = "GetTheTruth";
string methodDispIdName = "[DispID=1030]";
string s1 = (string)type.InvokeMember(methodName, BindingFlags.InvokeMethod, null, instance, null);
Console.WriteLine("Via name: 0", s1);
string s2 = (string)type.InvokeMember(methodDispIdName, BindingFlags.InvokeMethod, null, instance, null);
Console.WriteLine("Via DispID: 0", s2);
第一次调用成功,应用程序打印“Via name: I am King!”。
但是在第二次调用时(使用 DispID)我得到一个异常:
未处理的异常:System.MissingMethodException:找不到方法“TestComObject.AllKnowing.[DispID=1030]”。
在 OLE-COM 对象查看器中查看类型库时,我觉得它没问题:
// Generated .IDL file (by the OLE/COM Object Viewer)
//
// typelib filename: TestComObject.tlb
[
uuid(139DAA99-BF76-4057-BCC8-DCB35C153920),
version(1.0),
custom(90883F05-3D28-11D2-8F17-00A0C9A6186D, "TestComObject, Version=1.0.0.0, Culture=neutral, PublicKeyToken=58b6bd22fbd98427")
]
library TestComObject
// TLib : // TLib : mscorlib.dll : BED7F4EA-1A96-11D2-8F08-00A0C9A6186D
importlib("mscorlib.tlb");
// TLib : OLE Automation : 00020430-0000-0000-C000-000000000046
importlib("stdole2.tlb");
// Forward declare all types defined in this typelib
dispinterface IKnowEverything;
[
uuid(99476C04-574A-4271-ABC8-F1D2615C3D93),
version(1.0),
custom(0F21F359-AB84-41E8-9A78-36D110E6D2F9, "TestComObject.IKnowEverything")
]
dispinterface IKnowEverything
properties:
methods:
[id(0x00000406)]
BSTR GetTheTruth();
;
[
uuid(9B869BAF-622A-458B-BF80-2CD6D8A3C541),
version(1.0),
custom(0F21F359-AB84-41E8-9A78-36D110E6D2F9, "TestComObject.AllKnowing")
]
coclass AllKnowing
interface _Object;
[default] dispinterface IKnowEverything;
;
;
我在这里做错了什么?
【问题讨论】:
来自 Hans Passant 的回答:social.msdn.microsoft.com/Forums/en-US/csharplanguage/thread/…,当使用“[DispID=1030]”作为成员名称时,您应该在从Type.GetTypeFromProgID()
检索到的类型实例上调用 InvokeMember
,不是Type.GetTypeFromCLSID
无论我通过 ProgID 还是 CLSID 返回的 Type
都应该是相同的,不是吗?
我是这么认为的,但是 GetTypeFromProgID 的文档似乎说返回值始终是System.__ComObject
,而其他方法的文档没有这样说。当然,这不会是框架文档中出现不一致或错误的第一个实例。由于汉斯对您的问题给出了不同的答案,我想您对返回的Type
是正确的。
有趣,我会试试看是否有区别,如果没有提交关于 MSDN 文档的错误报告,那么我猜 ;)
【参考方案1】:
这不起作用,因为您在 .NET 中实现了 COM 服务器。 instance 成员是一个实际的 .NET 对象,而不是 RCW。从调试器可以看出,它不是 System.__ComObject 而是实际 .NET 类型的对象。
因此,当您使用 InvokeMember() 时,您将获得它的 System.Reflection 版本,而不是 IDispatch 版本。另一种查看方式是使用“getTheTruth”。注意第一次尝试现在也失败了。 IDispatch 不区分大小写,System.Reflection 不区分大小写。 “[DispID=1030]”技巧不起作用,因为 System.Reflection 对 disp-ids 一无所知。
功能,不是错误,当 CLR 可以直接创建 .NET 对象时,创建 RCW 是没有意义的。这也不容易实现。
【讨论】:
不知道这一点,感谢您提供的信息。我猜这在某些方面是有道理的。以上是关于如何从 .NET 对 .NET 中定义的 COM 对象执行后期绑定调用(通过 DispID)?的主要内容,如果未能解决你的问题,请参考以下文章
如何在 .NET 中使用自定义格式对 TimeSpan 对象进行 String.Format ?
如何在 JSON.NET 中实现自定义 JsonConverter?