为了账号安全,请及时绑定邮箱和手机立即绑定

是否可以覆盖非虚拟方法?

是否可以覆盖非虚拟方法?

C#
天涯尽头无女友 2019-11-15 12:59:38
有什么方法可以覆盖非虚拟方法?或产生类似结果的东西(除了创建一个新方法来调用所需的方法之外)?我想Microsoft.Xna.Framework.Graphics.GraphicsDevice在考虑单元测试的情况下覆盖其中的一种方法。
查看完整描述

3 回答

?
慕桂英546537

TA贡献1848条经验 获得超10个赞

不可以,您不能覆盖非虚拟方法。您可以做的最接近的事情是通过创建一个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方法都调用print Foo.M。如您所见,只要对该对象的引用具有正确的派生类型,但此方法的确可以为该方法提供新的实现,但是隐藏基本方法确实会破坏多态性。


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


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


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


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


如果要b.Foo在这种情况下进行调用,则CLR将正确确定b引用所指向的对象的类型,Bar并将调用M适当地分派给该对象。


查看完整回答
反对 回复 2019-11-15
?
慕村225694

TA贡献1880条经验 获得超4个赞

不,你不能。


您只能覆盖虚拟方法-在此处查看MSDN:


在C#中,派生类可以包含与基类方法同名的方法。


基类方法必须虚拟定义。


查看完整回答
反对 回复 2019-11-15
?
鸿蒙传说

TA贡献1865条经验 获得超7个赞

我认为您正在重载并覆盖混乱,重载意味着您拥有两个或更多个具有相同名称但参数集不同的方法,而重载意味着您对派生类中的方法具有不同的实现(从而替换或修改了行为在它的基类中)。

如果方法是虚拟的,则可以使用派生类中的override关键字覆盖它。但是,非虚拟方法只能通过使用new关键字代替override关键字来隐藏基本实现。如果调用者通过类型为基本类型的变量访问方法,则非虚拟路由将无用,因为编译器将对基方法使用静态分派(这意味着将永远不会调用派生类中的代码)。

从来没有任何阻止您向现有类添加重载的方法,但是只有知道您的类的代码才能访问它。


查看完整回答
反对 回复 2019-11-15
  • 3 回答
  • 0 关注
  • 356 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信