6

为什么以下两个代码示例会产生不同的输出?

情况1

enum EnumType
{
    First,
    Second,
    Third
}

class ClassB
{
    public string Func(int index)
    {
        return "Func(int)";
    }

    public string Func(EnumType type)
    {
        return "Func(EnumType)";
    }
}

class Program
{
    static void Main(string[] args)
    {
        ClassB b = new ClassB();
        Console.WriteLine(b.Func(0));
        Console.WriteLine(b.Func(EnumType.First));
        Console.ReadLine();
    }
}

输出:

Func(int)
Func(EnumType)

案例2

enum EnumType
{
    First,
    Second,
    Third
}

class ClassA
{
    public string Func(int index)
    {
        return "Func(int)";
    }
}

class ClassB : ClassA
{
    public string Func(EnumType enumType)
    {
        return "Func(EnumType)";
    }
}

class Program
{
    static void Main(string[] args)
    {
        ClassB b = new ClassB();
        Console.WriteLine(b.Func(0));
        Console.WriteLine(b.Func(EnumType.First));
        Console.ReadLine();
    }
}

输出:

Func(EnumType)
Func(EnumType)

我很困惑。这是否意味着 Func(EnumType) 隐藏了在基中声明的 Func(int) ?如果是这种情况,那么为什么在第二种情况下将文字 0隐式转换为 EnumType 而没有警告?

编辑:

当我尝试时,还有更有趣的行为

Console.WriteLine(b.Func(0));         
Console.WriteLine(b.Func(1));
Console.WriteLine(b.Func(EnumType.First));

你猜输出应该是什么样子?

这里是:

 Func(EnumType)
 Func(int)
 Func(EnumType)

任何想法为什么 0 和 1 被区别对待?

编辑2:

事实证明,字面量 0 在 C# 中确实具有特殊含义。

在这里这里,我找到了对这种行为的出色描述(请参阅已接受的答案)。

4

2 回答 2

5

是的,它确实隐藏了 A 类中声明的 Func(int)。

此外,请参阅枚举(C# 参考)

枚举元素的默认底层类型是 int

您可能还想看看多态性(C# 编程指南)

编辑

如果你要改变

Console.WriteLine(b.Func(0));          
Console.WriteLine(b.Func(1)); 
Console.WriteLine(b.Func(EnumType.First)); 

int i = 0;
Console.WriteLine(b.Func(i));          
Console.WriteLine(b.Func(1)); 
Console.WriteLine(b.Func(EnumType.First)); 

你会发现输出应该是

Func(int)   
Func(int)   
Func(EnumType)

如果直接传递给函数调用,似乎 0 会隐式转换为默认枚举值。

编辑 2

我检查了 IL 代码,它似乎确实将 0 隐式转换为枚举

IL_0000: nop
IL_0001: newobj instance void ClassB::.ctor()
IL_0006: stloc.0
IL_0007: ldloc.0
IL_0008: ldc.i4.0
IL_0009: callvirt instance string ClassB::Func(valuetype EnumType)
IL_000e: call void [mscorlib]System.Console::WriteLine(string)
IL_0013: nop
IL_0014: ldloc.0
IL_0015: ldc.i4.0
IL_0016: callvirt instance string ClassB::Func(valuetype EnumType)
IL_001b: call void [mscorlib]System.Console::WriteLine(string)
IL_0020: nop
IL_0021: call string [mscorlib]System.Console::ReadLine()
IL_0026: pop
IL_0027: ret
于 2012-07-17T04:33:06.717 回答
0

枚举的底层类型是 int,因此 ClassB 函数确实隐藏了 ClassA 版本。

尝试将其更改为:

enum EnumType : byte
{
    First,
    Second,
    Third
}
...
Console.WriteLine(b.Func(256));  // out of range for a byte

您应该看到它调用了 int 函数。

于 2012-07-17T04:35:26.777 回答