如何从 .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 ?

如何阻止对任何文件或 exe vb.net 的访问?

如何在 JSON.NET 中实现自定义 JsonConverter?

在 asp.net 中,如何从 usercontrol 获取自定义 webcontrol 的引用?

如何从.Net调用win32 CreateMutex

如何在vb6中注册一个不能用作com互操作的.net DLL?