C#:通用方法不调用特定方法重载

时间:2014-09-05 12:02:59

标签: c# generics mono unity3d polymorphism

我正在尝试在C#中创建一个泛型方法,它将根据其主体中的参数数据类型调用不同的方法,然后处理它们的结果。我试图通过创建一个通用的包装器方法来实现这一点,然后提供处理方法的几个重载 - 包括在没有特定的重载可用的情况下使用的通用方法。

当我直接调用处理方法时,正确选择了适当的版本。但是当我从包装器方法调用它时,它总是选择泛型方法,即使我传递给它的特定数据类型存在匹配的重载。

有没有办法调整代码使其行为符合我的需要?或者我必须使用不同的方法。

我需要代码与Mono 2.6兼容。

using System;

class Program
{
    static void Func<T>(T val)
    {
        Console.WriteLine("Generic Func");
    }

    static void Func(int val)
    {
        Console.WriteLine("Int Func");
    }

    static void Func(string val)
    {
        Console.WriteLine("String Func");
    }

    static void FuncWrap<T>(T val)
    {
        Console.Write("Wrap: ");
        Func(val);
    }

    static void Main(string[] args)
    {
        Func(2);
        Func("Potato");
        Func(2.0);

        FuncWrap(2);
        FuncWrap("Potato");
        FuncWrap(2.0);

        Console.Read();
    }
}

1 个答案:

答案 0 :(得分:14)

  

有没有办法纠正这种行为?

根据C#语言规范,它已经是正确的行为。在Func内调用的FuncWrap的重载通常在编译时确定 ,因此它无法根据执行时类型选择不同的Func重载

然而,改变行为的一种方法是使用动态类型:

static void FuncWrap<T>(T val)
{
    Console.Write("Wrap: ");
    dynamic x = val;
    Func(x);
}

现在,它将根据x值的实际类型在执行时执行重载解析。这会产生性能成本,但应该按照您的意愿行事。

或者,您可以对过载的知识进行硬编码:

static void FuncWrap<T>(T val)
{
    Console.Write("Wrap: ");
    if (typeof(T) == typeof(string))
    {
        Func((string)(object)val);
    }
    else if (typeof(T) == typeof(int))
    {
        Func((int)(object)val);
    }
    else
    {
        Func(val);
    }
}

但这显然非常可怕。