2

我有一个用于一些简单数学实用程序的 COM 对象。其中,它导出 2 个接口 - IDL 如下:

interface IUnivariateFunction : IDispatch
{
    HRESULT evaluate([in] double a, [out, retval] double* b);
};

interface IOneDimSolver : IDispatch
{
    HRESULT Solve([in] double min, [in] double max, [in] double targetAccuracy, [in] LONG maxIterations, [in] IUnivariateFunction* function, [out, retval] double* result);
};

然后一个实现IOneDimSolver是:

coclass Brent
{
    [default] interface IOneDimSolver;
};

如果我想在 C# 中使用这个对象并且我有可用的 TypeLibrary,那么使用这个功能非常容易——我可以实现一些函数来解决:

public class CalculatePi : IUnivariateFunction
{
    public double evaluate(double x)
    {
        return x - Math.PI;
    }
}

然后使用它:

var brent = new Brent();
var result = brent.Solve(-10.0, 10.0, 1E-10, 100, new CalculatePi());

这工作得很好,所以没有问题。但是,我还想证明我可以将它与后期绑定一起使用(此时几乎纯粹用于学术目的)。

var brent = Activator.CreateInstance(Type.GetTypeFromCLSID(new Guid("xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx")));
var result = brent.GetType().InvokeMember("Solve", BindingFlags.InvokeMethod, null, brent, new object[] { -10.0, 10.0, 1E-10, 100, new CalculatePi() });

看起来brent对象正在创建正常,但调用InvokeMember失败并显示消息{"Specified cast is not valid."}

我猜该类型CalculatePi与 COM IDispatch 机器所期望的不兼容,但我只是不确定如何使其工作。任何帮助都会很棒!

4

1 回答 1

3

{“指定的演员表无效。”}

这是一条 CLR 异常消息,由 InvalidCastException 产生。这极大地限制了这种事故的可能原因,您知道实际上并不是本机代码炸弹。在后期绑定代码中没有很多可能的原因。我只能想到一个。

最可能的原因是CalculatePi 类缺少该[ComVisible(true)]属性。因此,当 CLR 试图从对象中获取 IDispatch 接口指针时,它就会崩溃。它在早期绑定的情况下工作,C# 编译器可以从类型库中得知它需要生成 IUnivariateFunction 引用。这是可见的,因为它来自互操作库。

应该解决问题。但请记住, IOneDimSolver::Solve() 方法实际上与喜欢使用后期绑定的代码类型不兼容。脚本语言不知道如何实现 IUnivariateFunction 接口,它对它一无所知。您应该声明该参数IDispatch*,以便任何人都可以调用它。在实现中,使用QueryInterface() 获取IUnivariateFunction 接口指针。当它成功时,您会很高兴,让您可以快速轻松地拨打电话。但是当它没有时必须回退到 IDispatch::Invoke() 。如果您不想编写该代码,那么最好不要承诺 IDispatch 支持并从 IUnknown 派生接口。

于 2017-11-27T11:49:28.143 回答