为什么我不能在泛型类型参数受限于接口时使用'as'关键字?

23
在下面的示例中(仅用于演示目的),如果T未与类一起约束,则此转换:
var ret = objectA as T;

如果您这样做,将会导致以下编译错误:

类型参数“T”不能与“as”运算符一起使用,因为它没有类类型约束或“class”约束。

我不明白为什么我不能这样做。既然我已经将 T 约束为接口 IObject,编译器应该知道 T 必须是一个接口类型,而 as 操作应该是有效的。

public interface IObject
{
    string Id { get; set; }
}
public class ObjectA : IObject
{
    public string Id { get; set; }
}
public class ObjectFactory
{
    public T CreateObject<T>(string id) where T : IObject
    {
        ObjectA objectA = new ObjectA();
        var x = objectA as IObject; // this is good
        var ret = objectA as T; // why this 'as' cannot compile?
        return ret;
    }
    public T CreateClassObject<T>(string id) where T : class, IObject
    {
        ObjectA objectA = new ObjectA();
        var ret = objectA as T; // if T is class, this 'as' can compile
        return ret;
    }
}
1个回答

34

因为我已经将T限制为接口IObject,编译器应该知道T必须是接口类型,'as'操作应该是有效的。

不,T不一定是接口类型。它必须是实现该接口的类型。例如:

public struct Foo : IObject
{
    public string Id { get; set; }
}

现在你期望的是 CreateObject<Foo>("ff") 做什么?
由于在 CreateObject 上有约束 class,所以该调用将无效,因为 Foo 不是引用类型 - 编译器知道 T 是引用类型,所以 objectA as T 可行。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接