有什么方法可以覆盖非虚拟方法?或产生类似结果的东西(除了创建一个新方法来调用所需的方法之外)?

我想在考虑单元测试的情况下重写Microsoft.Xna.Framework.Graphics.GraphicsDevice中的方法。

最佳答案

不可以,您不能覆盖非虚拟方法。您可以做的最接近的事情是通过创建一个具有相同名称的new方法来隐藏该方法,但这是不可取的,因为它违反了良好的设计原则。

但是,即使隐藏方法也不会像真正的虚拟方法调用那样为您带来执行时间。考虑以下示例:

using System;

class Example
{
    static void Main()
    {
        Foo f = new Foo();
        f.M();

        Foo b = new Bar();
        b.M();
    }
}

class Foo
{
    public void M()
    {
        Console.WriteLine("Foo.M");
    }
}

class Bar : Foo
{
    public new void M()
    {
        Console.WriteLine("Bar.M");
    }
}

在此示例中,对M方法的两个调用都打印Foo.M。正如您所看到的,只要对该对象的引用具有正确的派生类型,但隐藏该基本方法的确会破坏多态性,因此该方法的确为该方法提供了新的实现。

我建议您不要以这种方式隐藏基本方法。

我倾向于那些支持C#默认行为的人,即默认情况下方法是非虚拟的(与Java相对)。我会更进一步地说,默认情况下也应该密封类。很难正确设计继承,并且存在一个未标记为虚拟的方法的事实表明该方法的作者从未打算重写该方法。

编辑:“执行时间多态调度” :

我的意思是调用虚拟方法时在执行时发生的默认行为。举例来说,在我之前的代码示例中,实际上没有定义虚拟方法,而是定义了虚拟方法,而没有定义非虚拟方法。

如果在这种情况下我要调用b.Foo,则CLR将正确地确定b引用指向的对象的类型为Bar,并适当地将调用分派(dispatch)给M

09-30 15:37
查看更多