隐式运算符使用接口

Mic*_*ows 53 c# compiler-construction generics casting implicit-conversion

我有一个泛型类,我正在尝试实现隐式类型转换.虽然它主要起作用,但它不适用于界面转换.经过进一步调查,我发现存在编译器错误:"来自接口的用户定义转换"适用.虽然我知道在某些情况下应该强制执行,但我正在尝试做的事情似乎是合法的案例.

这是一个例子:

public class Foo<T> where T : IBar
{
    private readonly T instance;

    public Foo(T instance)
    {
        this.instance = instance;
    }
    public T Instance
    {
        get { return instance; }
    }
    public static implicit operator Foo<T>(T instance)
    {
        return new Foo<T>(instance);
    }
}
Run Code Online (Sandbox Code Playgroud)

使用它的代码:

var concreteReferenceToBar = new ConcreteBar();
IBar intefaceReferenceToBar = concreteReferenceToBar;
Foo<ConcreteBar> concreteFooFromConcreteBar = concreteReferenceToBar;
Foo<IBar> fooFromConcreteBar = concreteReferenceToBar;
Foo<IBar> fooFromInterfaceBar = intefaceReferenceToBar; // doesn't work
Run Code Online (Sandbox Code Playgroud)

有没有人知道一个解决方法,或者任何人都能以令人满意的方式解释为什么我不能施展 interfaceReferenceToBar隐式地Foo<IBar>,因为在我的情况下它没有被转换,只包含在Foo中?

编辑: 看起来协方差可能提供救赎.我们希望C#4.0规范允许使用协方差隐式转换接口类型.

Ada*_*hes 57

你不能这样做的原因是因为它在C#语言规范中是特别禁止的:

如果满足以下所有条件,则允许类或结构声明从源类型S到目标类型T的转换:

  • ...
  • S和T都不是object接口类型.

不允许用户定义的转换从接口类型转换或转换为 接口类型.特别是,此限制确保在转换为接口类型时不会发生用户定义的转换,并且只有在转换 的对象实际实现指定的接口类型时,才能成功转换为 接口类型.

资源

  • 我*相信*对隐式接口转换的限制与如何实现COM互操作有关.COM使用QueryInterface,.NET自动处理.允许隐式接口转换会产生干扰. (3认同)
  • @MichaelMeadows,你可能会开心阅读Adam Houldsworth和Eric Lippert关于[我的类似问题]的回复(http://stackoverflow.com/questions/9229928/more-on-implicit-conversion-operators-and-interfaces-在C#4.0上下文中再次使用. (3认同)
  • 我知道这是规范的一部分,接口的隐式转换在某些情况下应该是无效的,但在所有情况下? (2认同)
  • 我同意你的观点,我不知道为什么他们让它对所有情况都无效。在这种情况下,您可以在编译时确定强制转换(应该)有效。 (2认同)
  • 它是否解释了为什么他们在任何地方做出这个决定? (2认同)