C#继承中的override(重写)与new(覆盖)用法

  刚接触C#编程,我也是被override与new搞得晕头转向。于是花了点时间翻资料,看博客,终于算小有领悟,把学习笔记记录于此。

  首先声明一个父类Animal类,与继承Animal的两个子类Dog类与Cat类。父类Animal中有一个Say方法,而子类Dog与Cat分别override(重写)与new(覆盖)了Say方法。

  让我们通过实例来看一下这两者有什么相同与不同之处。

 1     public class Animal
 2     {
 3         public virtual void Say()
 4         {
 5             Console.WriteLine("Hello,Animal");
 6         }
 7     }
 8     public class Dog : Animal
 9     {
10         public override void Say()
11         {
12             Console.WriteLine("Hello,Dog");
13         }
14     }
15     public class Cat : Animal
16     {
17         public new void Say()
18         {
19             Console.WriteLine("Hello,Cat");
20         }
21     }

首先说override与new的共同点:

  1. 都是子类在重新实现父类中签名相同的方法时使用。
  2. 当声明一个子类对象时,调用这个方法,调用的都是子类中实现的方法。

    例如:

 1 class Program
 2     {
 3         static void Main(string[] arge)
 4         {
 5             Dog d = new Dog();
 6             Cat c = new Cat();
 7             d.Say();//调用override的方法
 8             c.Say();//调用new的方法
 9         }
10     }

   输出是:

Hello,Dog
Hello,Cat

    此时调用的分别是Dog与Cat类中实现的Say方法。

  3.都不会影响父类自身的方法。

   如:

1 class Program
2     {
3         static void Main(string[] arge)
4         {
5             Animal a = new Animal();
6             a.Say();//调用父类方法。未受影响。
7         }
8     }

  此时的输出是:

Hello,Animal

下面说两者的不同之处:

  1.

  (1)override:父类方法必须用virtual修饰,表示这个方法是虚方法,可以被重写。否则不能被重写。

  (2)new :   父类方法不必使用virtual修饰。

  2.

  (1)override : 使用override时,父类中必须存在签名完全相同的virtual方法。否则编译不通过。

    如果我在Dog类的Say增加一个string类型的形参,则编译器会提示:没有合适的方法可以重写。

  (2)new :   使用new时,父类中最好存在签名相同的方法。如果没有,VS会有提示,但编译不会报错。此时,new关键字便没有了意义。

   如果我在Cat类的Say增加一个string类型的形参,VS会提示:new关键字不是必须的。

  3.当子类中存在与父类方法签名相同的方法,而没有被override或new修饰时,默认为new。

  也就是说,override必须写,而new可以不写(不推荐)。

  4.这是最重要的一点。以上三点都是使用方法的区别,而这一点是两者在实际使用时效果的区别。

  (1)override :重写后,当子类对象转换为父类时,无法访问被重写的虚方法。也就是,被子类重写后,虚方法在这个子类中便失效了。

   如:

class Program
    {
        static void Main(string[] arge)
        {
            Dog d = new Dog();
            Animal a = d as Animal;//子类转换为父类。注意此时a与d指向同一对象,但d是作为Dog类访问,而a是作为Animal类访问
            d.Say();//此时调用的是override的方法
            a.Say();//此时调用的也是override的方法
        }
    }

    输出为:

Hello,Dog
Hello,Dog

  两次调用的都是Dog中重写的Say方法

  (2)new : 覆盖后,当子类对象转换为父类,可以访问被覆盖的父类方法。也就是,转换为父类后,子类new的方法便失效了,此时调用的是父类方法。
       当其再转换为子类时,调用的又变为子类方法。

    如:

 1 class Program
 2     {
 3         static void Main(string[] arge)
 4         {
 5             Cat c = new Cat();
 6             Animal a = c as Animal;//子类转换为父类。注意此时a与c指向同一对象,但c是作为Cat类访问,而a是作为Animal类访问
 7             c.Say();//此时调用的是new的方法
 8             a.Say();//此时调用的是父类中的方法
 9         }
10     }

   此时的输出为:

Hello,Cat
Hello,Animal

内存原理:
  我们都知道,调用对象的方法,实际上是访问对象方法在内存中的地址。那么既然可以通过c.Say()访问到父类的方法,说明在对象c中,也有Animal类的Say方法。
  事实上,当子类继承父类的时候,父类中所有方法、字段(包括私有的)都会复制一份放在子类中。而我们所谓的重写和覆盖,重写、覆盖的是存放在子类中的,复制出来的方法,而不是父类中的方法,所以当然不会对父类产生任何影响。而不能调用私有的、或者被重写的方法或字段,是由于无权访问,而不是内存中不存在。

时间: 2025-01-17 18:54:21

C#继承中的override(重写)与new(覆盖)用法的相关文章

C#继承中abstract、virtual、override和new

abstract 详细参考:https://msdn.microsoft.com/zh-cn/library/sf985hc5.aspx abstract 修饰符指示所修饰的内容缺少实现或未完全实现. abstract 修饰符可用于类.方法.属性.索引器和事件. 在类声明中使用 abstract 修饰符以指示某个类只能是其他类的基类. 标记为抽象或包含在抽象类中的成员必须通过从抽象类派生的类来实现. 抽象类具有以下特性: - 抽象类不能实例化. - 抽象类可以包含抽象方法和抽象访问器. - 不能

day4(继承、修饰符、方法的重写、super关键词、继承中的构造方法、object类、对象转型)

1.类的继承(extends)通过继承,子类自动拥有父类的所有成员(成员变量和成员发方法)一个子类只能有一个父类,一个父类可以派生多个子类 2.访问修饰符(private 默认 protected public )对于class的权限修饰只可以用public和defaultdefault只能被同一个包内部的类访问 3.方法的重写(override)在子类中可以根据需要对从父类中继承来的方法进行重写重写方法必须和被重写方法具有相同方法名称.参数列表和返回类型重写方法不能使用比被重写方法更严格的访问

【转载】 C++多继承中重写不同基类中相同原型的虚函数

本篇随笔为转载,原文地址:C++多继承中重写不同基类中相同原型的虚函数. 在C++多继承体系当中,在派生类中可以重写不同基类中的虚函数.下面就是一个例子: class CBaseA { public: virtual void TestA(); }; class CBaseB { public: virtual void TestB(); }; class CDerived : public CBaseA, public CBaseB { public: virtual void TestA()

Java多态机制和继承中重写重载

关于Java中多态机制 http://www.cnblogs.com/chenssy/p/3372798.html 这篇博文讲的很透彻 大体意思是 多态定义: 多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定.因为在程序运行时才确定具体的类,这样,不用修改源程序代码,就可以让引用变量绑定到各种不同的类实现上

在继承中重写方法时抛出异常的问题

要清楚的知道子类重写父类的方法时,当父类抛出异常时,子类抛出的异常需要怎么写,首先必须了解异常的类型. Exception 这种异常分两大类运行时异常和非运行时异常(编译异常). 运行时异常:都是RuntimeException类及其子类异常,如NullPointerException(空指针异常).IndexOutOfBoundsException(下标越界异常)等,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理.这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类

继承、继承中的重写

[修饰符] class 子类名 extends 父类名 { 类体 } 例: public class Bird{//父类 String color="white"; String skin="羽毛"; } public class Pigeon extends Bird{//子类 public static void main(String[] args) { Pigeon pigeon=new Pigeon();//建立类的对象 System.out.printl

Java中的Overload(重载)与Override(重写、覆盖)

java中的方法重载发生在同一个类中两个或者多个方法的方法名相同但是参数不同的情况,方法重载是指子类与父类之间子类重新定义了父类的方法,重写的方法与原方法签名.返回值.参数完全相同.Overload(重载) 重载Overload是一个类中多态性的一种表现,是编译时的多态性.方法重载是让类以统一的方式处理不同类型数据的一种手段.重载发生在同一个类中.同名的方法如果有不同的参数列表(参数个数.类型)则视为重载. java方法的重载,就是可以在类中创建多个方法,这些方法具有相同的名字,但具有不同的参数

Java中的方法重写 与 final关键字

Java中的方法重写 与 final关键字 以下程序是否存在错误? class Base { private final void method() { System.out.println("In Base..."); } } class Sub extends Base { public void method() { System.out.println("In Sub..."); } } 答案 正确的. 使用final关键字可以声明类.成员变量和成员方法,一

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(