专栏首页TheOneGIS空间站C#中的override和new关键字

C#中的override和new关键字

问题1: 虚方法

首先,看下面的代码。

namespace CSharpTest
{
    class A
    {
        public void fun()
        {
            Console.WriteLine("这是父类方法");
        }
    }

    class B : A
    {
        public void fun()
        {
            Console.WriteLine("这是子类方法");
        }
    }

    class Test
    {
        public static void Main()
        {
            A a = new A();
            a.fun();
            a = new B();
            a.fun();
        }
    }
}

猜猜结果是什么? 如果对Java熟悉的朋友,可能会认为是结果:

这是父类方法
这是子类方法

但是其实运行结果是:

这是父类方法
这是父类方法

这是因为Java中的类方法默认是虚函数(虽然Java中没有这个叫法),子类函数会默认覆盖父类的同名函数(Java后来提供了@override注解)。然而C#中必须使用virtual关键字显示声明该函数是虚函数,然后在子类中使用override关键字重写父类方法,这才真正实现了对父类方法的重写,才能实现多态(C++中的多态就是使用虚函数实现的,而且和C#一样必须使用virtual关键字显示声明)。

所以,要输出结果为:

这是父类方法
这是子类方法

需要修改代码如下:

namespace CSharpTest
{
    class A
    {
        public virtual void fun()
        {
            Console.WriteLine("这是父类方法");
        }
    }

    class B : A
    {
        public override void fun()
        {
            Console.WriteLine("这是子类方法");
        }
    }

    class Test
    {
        public static void Main()
        {
            A a = new A();
            a.fun();
            a = new B();
            a.fun();
        }
    }
}

即给父类的方法添加virtual关键字,给子类的同名方法添加override关键字。


问题2:override和new关键字

首先看下面的代码:

namespace CSharpTest
{
    class C
    {
        public virtual void fun()
        {
            Console.WriteLine("这是一个虚方法");
        }
    }
    class C1 : C
    {
        public override void fun()
        {
            Console.WriteLine("使用override关键字修饰的方法");
        }
    }
    class C2 : C
    {
        public new void fun()
        {
            Console.WriteLine("使用new关键字修饰的方法");
        }
    }

    class Test
    {
        public static void Main()
        {
            C c1 = new C1();
            c1.fun();
            C c2 = new C2();
            c2.fun();
        }
    }
}

猜猜上面的代码输出结果? 正确的结果是:

使用override关键字修饰的方法
这是一个虚方法

为什么使用关键字new修饰的方法,调用的是父类的方法呢? 是不是很奇怪?为什么使用override关键字的子类方法被调用了,而使用new关键字的子类方法没有被调用。 首先看看override关键字:override方法为从基类继承的成员提供新的实现。以override声明重写的方法被称为被重写的基类方法,被重写的基类方法必须具有与重写方法相同的签名。非虚方法或者静态方法不能被重写,被重写的基类必须是virtual、abstract或者override的。override声明不能改变虚方法的可访问性,override方法和virtual方法必须具有相同的访问级别修饰符。不能使用下列修饰符修饰重写方法:new、static、virtual和abstract。 new关键字:new修饰符用来明确地隐藏由基类继承而来的成员。要隐藏继承而来的成员,可以在派生类中共用相同的名称并用new修饰符修饰它。 下面来分析我们的程序: c1.fun();因为子类C1使用override关键字重写了父类的方法,基类C和子类C1都具有fun()方法,所以c1.fun()会动态调用C1的fun()方法而不是父类的。 c2.fun();子类C2使用new关键字隐藏了父类的方法,相当于子类中的fun()方法是直接继承自父类的。而子类中使用new关键字声明的fun()方法是另一个方法,只是恰巧与子类的fun()方法同名而已(是不是有些糊涂了)。所以 c2.fun()会调用父类的fun()方法,要想调用C2的fun()方法必须吧c2强制转换为C2.


下面看看微软官方的文档解释: C# 语言经过专门设计,以便不同库中的基类与派生类之间的版本控制可以不断向前发展,同时保持向后兼容。这具有多方面的意义。例如,这意味着在基类中引入与派生类中的某个成员具有相同名称的新成员在 C# 中是完全支持的,不会导致意外行为。它还意味着类必须显式声明某方法是要重写一个继承方法,还是一个隐藏具有类似名称的继承方法的新方法。 在 C# 中,派生类可以包含与基类方法同名的方法。 基类方法必须定义为 virtual。 如果派生类中的方法前面没有 new 或 override 关键字,则编译器将发出警告,该方法将有如存在 new 关键字一样执行操作。 如果派生类中的方法前面带有 new 关键字,则该方法被定义为独立于基类中的方法。 如果派生类中的方法前面带有 override 关键字,则派生类的对象将调用该方法,而不是调用基类方法。 可以从派生类中使用 base 关键字调用基类方法。 override、virtual 和 new 关键字还可以用于属性、索引器和事件中。 默认情况下,C# 方法为非虚方法。如果某个方法被声明为虚方法,则继承该方法的任何类都可以实现它自己的版本。若要使方法成为虚方法,必须在基类的方法声明中使用 virtual 修饰符。然后,派生类可以使用 override 关键字重写基虚方法,或使用 new 关键字隐藏基类中的虚方法。如果 override 关键字和 new 关键字均未指定,编译器将发出警告,并且派生类中的方法将隐藏基类中的方法。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • RecyclerView零点突破(基本使用篇)

    张风捷特烈
  • GuavaCache学习笔记二:Java四大引用类型回顾

    上一篇已经讲了,如何自己实现一个LRU算法。但是那种只是最基本的实现了LRU的剔除策略,并不能在生产中去使用。因为Guava Cache中使用的是SoftRef...

    一枝花算不算浪漫
  • 【精编重制版】JavaWeb 入门级项目实战 -- 文章发布系统 (第三节)

    【精编重制版】JavaWeb 入门级项目实战 -- 文章发布系统 (第三节) - Java520官方.png

    剽悍一小兔
  • 十年Java架构师大厂实战(年薪80W)

    BAT一直是程序员心神向往的地方,那些最最前沿的一线互联网技术都出自这些合称为大厂的地方,那里人才济济,哪怕实战经验不好的程序员进了那里,都会受到技术的熏陶,培...

    美的让人心动
  • 为什么程序员要用两个大屏幕?是为了炫耀吗?

    一墨编程学习
  • 一分钟上手Rinetd——端口转发工具 原

    iptables 的功能当然强大,但理解与设置却有点抽象,便通过google认识了rinetd。

    wuweixiang
  • 利用TurniBit开发板DIY一套自动窗帘模拟系统

    步进电机是将电脉冲信号转变为角位移或线位移的开环控制电机,是现代数字程序控制系统中的主要执行元件,应用极为广泛。在非超载的情况下,电机的转速、停止的位置只取决于...

    周俊辉
  • LeetCode 26 Remove Duplicates from Sorted Array

    一份执着✘
  • 剑指offer 把数组排成最小的数

    输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。例如输入数组{3,32,321},则打印出这三个数字能排成的最小数字...

    week
  • 剑指offer 翻转单词顺序列

    牛客最近来了一个新员工Fish,每天早晨总是会拿着一本英文杂志,写些句子在本子上。同事Cat对Fish写的内容颇感兴趣,有一天他向Fish借来翻看,但却读不懂它...

    week

扫码关注云+社区

领取腾讯云代金券