2

我使用TlbImp.exe. 一个有一个包装器描述为的类

using System;
using System.Runtime.InteropServices;

namespace GNOTDRSIGNATURESERVERLib
{
    [CoClass(typeof(GNOTDRSignatureServerClass)),
        Guid("20CBF9E0-06BF-11D3-97B5-0080C878CFFA")]
    public interface GNOTDRSignatureServer
    {
    }
}

该对象必须作为参数传递

using System;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;

namespace GN8000Lib
{
    [TypeLibType(2),
         ClassInterface(0), 
         ComSourceInterfaces("GN8000Lib._I8000Events\0"),
         Guid("DCAD84FE-43A8-11D3-B1A2-005004131886")]
    public class GN8000Class : I8000
    {
        [MethodImpl(MethodImplOptions.InternalCall)]
        public extern GN8000Class();

        [DispId(6)]
        [MethodImpl(MethodImplOptions.InternalCall)]
        void I8000.Initialize(
            [IUnknownConstant] [MarshalAs(UnmanagedType.IUnknown)] [In]
            object pSigServer = null);
    }
}

然后我这样调用方法:

var gn8000 = new GN8000();
var signatureServer = new GNOTDRSignatureServer();
gn8000.Initialize(signatureServer);

我认为正在传递的对象不是正确的。我觉得TlbImp.exe可以链接那些 DLL 并使用GNOTDRSignatureServer而不是UnmanagedType.IUnknown,或者我可以使用System.Runtime.InteropServices.Marshal.GetComInterfaceForObject

我的代码有什么问题?

4

1 回答 1

2

这些声明看起来很丑陋,我假设它们是由某种逆向工程工具生成的,而你不是自己编写的。在这种情况下,拥有一个空接口并不少见。COM 中的等价物是dispinterface,一个仅支持通过 IDispatch 进行后期绑定的接口。这是相当普遍的,COM 服务器可能被设计为使用脚本语言、只支持后期绑定的运行时环境。与 .NET 中的 ComInterfaceType.InterfaceIsIDispatch 相同。[DispId] 对于后期绑定很重要,通过号码而不是名称调用。

这也使得参数类型易于解释。脚本语言使用 VARIANT 作为其变量类型。与您在 C# 代码中声明为对象的变量非常相似,它可以存储任何值。功能,而不是错误。

于 2012-02-02T14:13:18.137 回答