我对以下场景感到困惑,我们有一组这样的类
public class A
{
public virtual string SomeMethod()
{
return "A";
}
}
public class B : A
{
public override string SomeMethod()
{
return "B";
}
}
public class C : B
{
public new virtual string SomeMethod()
{
return "C";
}
}
public class D : C
{
public override string SomeMethod()
{
return "D";
}
}
当我调用下面的方法时
private void Test()
{
A a = new D();
B b = new D();
C c = new D();
D d = new D();
Console.WriteLine(a.SomeMethod()); //The method in class B is being called
Console.WriteLine(b.SomeMethod()); //The method in class B is being called
Console.WriteLine(c.SomeMethod()); //The method in class D is being called
Console.WriteLine(d.SomeMethod()); //The method in class D is being called
}
我得到这样的输出
B B D D
为什么调用的是继承的方法,而不是声明类型中的方法,为什么不是每次都调用类D
中的方法?
最佳答案
当您调用 SomeMethod
时在 A
的实例上(即 A foo = new <A or derived>()
),您希望获得 SomeMethod
的最衍生版本可用的。这就是继承的全部要点。您的代码只需要处理基类的实例,但如果这些实例恰好属于派生类,则会调用特殊的派生实现。这是您重写方法时得到的行为。
这解释了常见的场景,例如
A b = new B();
b.SomeMethod(); // B's implementation invoked
与 new
, 你并没有覆盖这个方法,你是在声明一个全新的同名方法。 这个方法只存在于声明它的类及其子类中,它不是基类及其子类的一部分继承链。
那么 A a = new D(); a.SomeMethod();
是什么意思?做?该变量声明为 A
,因此必须在 A
上定义对其调用的任何方法/属性/等. C
定义了一个新方法SomeMethod
(并且 D
覆盖了它),但此方法在 A
上不存在,所以这里不能调用。 SomeMethod
的最派生实现(在类型 A
上声明)在 B
中,所以这是被调用的实现。
B b = new D(); b.SomeMethod();
同样的故事
只有当我们到达 C c = new D(); c.SomeMethod()
时new 方法有机会执行。这是因为变量是 C
,因此新方法SomeMethod
是第一次在变量类型上声明。如上所述,将调用派生程度最高的可能版本,在本例中这意味着被 D
覆盖的版本.
我希望我们都在同一页上 D d = new D(); d.SomeMethod()
:)
其他人发布了很好的链接。这是另一个可能有帮助的话题:C# and method hiding
这里的最后一个示例显示了一个更奇怪的场景,其中"new"方法被声明为私有(private),因此进一步派生的类型实际上继承了该方法的基本版本,而不是继承"new"版本。 http://msdn.microsoft.com/en-us/library/aa691135
关于c# - Method覆盖和方法隐藏的执行流程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12140423/