18

C# 4,为了简化 COM 互操作,允许 COM 接口的调用者在 by ref 参数的参数前省略 ref 关键字。

今天我惊讶地发现这也适用于扩展 COM 接口的扩展方法。请参阅以下,编译,代码:

using System;
using System.Runtime.InteropServices;

[ComImport, Guid ("cb4ac859-0589-483e-934d-b27845d5fe74")]
interface IFoo {
}

static class Program {

    public static void Bar (this IFoo self, ref Guid id)
    {
        id = Guid.NewGuid ();
    }

    static void Main ()
    {
        Foo (null);
    }

    static void Foo (IFoo o)
    {
        Guid g = Guid.NewGuid ();
        Console.WriteLine (g);

        // note that g is passed as is, and not as ref g    
        o.Bar (g);

        Console.WriteLine (g);
    }
}

我在规范中没有找到任何解释这种行为的东西。

我的感觉是 COM 接口之外的代码,即使它是扩展 COM 接口的扩展方法,也应该遵循常规的 C# 规则,并强制使用 ref 关键字。因此,我在 connect 上提交了一个错误。并不是说我认为这会被修复,即使它被认为是一个错误,也已经有依赖于此的代码。

漏洞?不是bug?

4

1 回答 1

2

我不认为这是一个错误。正如你所说,它看起来更像是“COM voodoo”。在底层,C# 编译器会发出一些实际上是正确的,如下所示:

private static void Foo(IFoo o)
{
    ...
    Guid g = Guid.NewGuid();
    Guid <>r__ComRefCallLocal0 = g;
    Bar(o, ref <>r__ComRefCallLocal0);
    ...
}

C# 实际上充满了技巧。如果您向 IFoo 添加一个方法,例如,像这样,

[ComImport, Guid("cb4ac859-0589-483e-934d-b27845d5fe74")]
interface IFoo
{
    void Test([Optional] ref object test);
}

再次,您将能够在 C# 4 中声明这一点:

static void Foo(IFoo o)
{
    Guid g = Guid.NewGuid();
    o.Test(g);
}

当然,这一切都只是因为 CSC.EXE 对 ComImport 属性有深入的了解。这些新的神奇互操作技巧已添加到 C# 4.0 中,以便能够轻松地与现有 COM 接口进行互操作。好吧,主要是针对 Microsoft Office 接口和方法,尤其是可怕的“缺少参考”参数的军队 :-)

我认为这在任何地方都没有完全指定。这就是 C# 4 规范的全部内容:

17.5 互操作属性 注意:本节仅适用于 C# 的 Microsoft .NET 实现。17.5.1 与 COM 和 Win32 组件的互操作 .NET 运行时提供了大量属性,使 C# 程序能够与使用 COM 和 Win32 DLL 编写的组件进行互操作。例如,DllImport 属性可用于静态 extern 方法,以指示该方法的实现可在 Win32 DLL 中找到。这些属性可在 System.Runtime.InteropServices 命名空间中找到,这些属性的详细文档可在 .NET 运行时文档中找到。

以下是 MSDN 上的一些页面:

于 2012-01-19T18:17:19.750 回答