6

我有一些扩展方法使用Expression参数来拉入属性成员并对其进行操作,并且对于成员是IEnumerable<>的特定情况,我有一个重载。但是,当从泛型类内部调用时(对于下面的 r4 ) ,它似乎与预期的方法重载不匹配。在类之外选择了正确的方法。

这里发生了什么?这会奏效还是我需要找到一种新方法?

(这是在 C# 5 中)

public class Test
{
    public void MyTest()
    {
        // returns "Object"
        var r1 = new MyClass<object>().Ext(a => a.Content);

        // returns "Enumerable"
        var r2 = new MyClass<IEnumerable<object>>().Ext(a => a.Content);

        // returns "Object"
        var r3 = new MyClass<object>().TestExt();

        // returns "Object" (I was expecting "Enumerable")
        var r4 = new MyClass<IEnumerable<object>>().TestExt();

        // returns "Enumerable"
        var r5 = new MyClass<int>().TestExt2();
    }
}

public class MyClass<T>
{
    public T Content { get; set; }

    public IEnumerable<object> OtherContent { get; set; }

    public string TestExt()
    {
        return this.Ext(a => a.Content);
    }

    public string TestExt2()
    {
        return this.Ext(a => a.OtherContent);
    }
}

public static class MyExtensions
{
    public static string Ext<T>(this T obj, Expression<Func<T, IEnumerable<object>>> memberExpression)
    {
        return "Enumerable";
    }

    public static string Ext<T>(this T obj, Expression<Func<T, object>> memberExpression)
    {
        return "Object";
    }
}
4

1 回答 1

4

泛型不是动态类型。要调用的重载在编译时被冻结。当程序稍后运行时,即使变量碰巧拥有更具体的运行时类型,也没有关系,因为重载是在编译时修复的。

你的方法:

public string TestExt()
{
    return this.Ext(a => a.Content);
}

必须在编译时绑定到Ext. 由于我们所知道T的类中的(类型a.ContentMyClass<T>是可转换为 的object,因此实际上只有一个重载可供选择,所以这对编译器来说很容易。

从那时起,TestExt方法体被硬编码为调用Ext.


编辑:这是一个更简单的例子:

static void Main()
{
    IEnumerable<object> e = new List<object>();
    var r = Generic(e);
}

static string Generic<T>(T x)
{
    return Overloaded(x);
}

static string Overloaded(IEnumerable<object> x)
{
    return "Enumerable";
}
static string Overloaded(object x)
{
    return "Object";
}

正如您现在所了解的那样,r变为"Object".

T(例如,如果您以某种方式进行约束where T : IEnumerable<object>,情况会有所不同)。

这对于运营商来说也是一样的。例如,==运算符在某种意义上是重载的,它以一种方式用于一般引用类型,而以另一种方式用于字符串。所以在下面的例子中, operator==扮演Overloadedfrom before 的角色:

static void Main()
{
    string str1 = "abc";
    string str2 = "a";
    str2 += "bc";        // makes sure this is a new instance of "abc"

    bool b1 = str1 == str2;        // true
    bool b2 = Generic(str1, str2); // false
}

static bool Generic<T>(T x, T y) where T : class
{
    return x == y;
}

b2变成false. _

于 2013-09-03T08:07:19.590 回答