https://msdn.microsoft.com/en-us/library/ms173153.aspx

In C#, a method in a derived class can have the same name as a method in the base class.

You can specify how the methods interact by using the new and override keywords.

The override modifier extends the base class method, and the new modifier hides it.

The difference is illustrated in the examples in this topic.

In a console application, declare the following two classes, BaseClass and DerivedClass.

DerivedClass inherits from BaseClass.

class BaseClass
{
public void Method1()
{
Console.WriteLine("Base - Method1");
}
} class DerivedClass : BaseClass
{
public void Method2()
{
Console.WriteLine("Derived - Method2");
}
}

In the Main method, declare variables bc, dc, and bcdc.

  • bc is of type BaseClass, and its value is of type BaseClass.

  • dc is of type DerivedClass, and its value is of type DerivedClass.

  • bcdc is of type BaseClass, and its value is of type DerivedClass. This is the variable to pay attention to.

Because bc and bcdc have type BaseClass, they can only directly access Method1, unless you use casting.

Variable dc can access both Method1 andMethod2. These relationships are shown in the following code.

class Program
{
static void Main(string[] args)
{
BaseClass bc = new BaseClass();
DerivedClass dc = new DerivedClass();
BaseClass bcdc = new DerivedClass(); bc.Method1();
dc.Method1();
dc.Method2();
bcdc.Method1(); //这个是正常的例子,子类没有重写父类的方法
}
// Output:
// Base - Method1
// Base - Method1
// Derived - Method2
// Base - Method1
}

Next, add the following Method2 method to BaseClass.

The signature of this method matches the signature of the Method2 method in DerivedClass.

public void Method2()
{
Console.WriteLine("Base - Method2");
}
 class BaseClass
{
public void Method1()
{
Console.WriteLine("Base - Method1");
}
public void Method2()
{
Console.WriteLine("Base - Method2");
}
} class DerivedClass : BaseClass
{
public void Method2()
{
Console.WriteLine("Derived - Method2");
}
}

父类有Method1和Method2    子类有Method2,并且子类的Method2隐藏了父类的Method2

Because BaseClass now has a Method2 method, a second calling statement can be added for BaseClass variables bc and bcdc, as shown in the following code.

bc.Method1();
bc.Method2();
dc.Method1(); //子类没有Method1,所以调用的是父类的方法
dc.Method2(); //子类有Method2,所以优先调用子类的方法
bcdc.Method1(); //子类没有Method1,所有调用的是父类的方法
bcdc.Method2(); //子类虽然有Method2,但是隐藏了父类Method2,

When you build the project, you see that the addition of the Method2 method in BaseClass causes a warning.

The warning says that the Method2 method in DerivedClass hides the Method2 method in BaseClass.

You are advised to use the new keyword in the Method2 definition if you intend to cause that result.

Alternatively, you could rename one of the Method2 methods to resolve the warning, but that is not always practical.

Before adding new, run the program to see the output produced by the additional calling statements. The following results are displayed.

// Output:
// Base - Method1
// Base - Method2
// Base - Method1
// Derived - Method2
// Base - Method1
// Base - Method2

The new keyword preserves the relationships that produce that output, but it suppresses抑制 the warning.

The variables that have type BaseClass continue to access the members of BaseClass,

and the variable that has type DerivedClass continues to access members in DerivedClass first, and then to consider members inherited from BaseClass.

To suppress the warning, add the new modifier to the definition of Method2 in DerivedClass, as shown in the following code.

The modifier can be added before or after public.

public new void Method2()
{
Console.WriteLine("Derived - Method2");
}

Run the program again to verify that the output has not changed.

Also verify that the warning no longer appears.

By using new, you are asserting断言 that you are aware that the member that it modifies hides a member that is inherited from the base class.

For more information about name hiding through inheritance, see new Modifier (C# Reference).

To contrast对比 this behavior to the effects of using override, add the following method to DerivedClass.

The override modifier can be added before or after public.

public override void Method1()
{
Console.WriteLine("Derived - Method1");
}

Add the virtual modifier to the definition of Method1 in BaseClass. The virtual modifier can be added before or after public.

public virtual void Method1()
{
Console.WriteLine("Base - Method1");
}

Run the project again. Notice especially the last two lines of the following output.

// Output:
// Base - Method1
// Base - Method2
// Derived - Method1
// Derived - Method2
// Derived - Method1
// Base - Method2

The use of the override modifier enables bcdc to access the Method1 method that is defined in DerivedClass.

Typically, that is the desired behavior in inheritance hierarchies.

You want objects that have values that are created from the derived class to use the methods that are defined in the derived class.

You achieve that behavior by using override to extend the base class method.

The following code contains the full example.

The following example illustrates similar behavior in a different context.

The example defines three classes: a base class named Car and two classes that are derived from it, ConvertibleCar有活动折篷的汽车 and Minivan小型货车.

The base class contains a DescribeCar method.

The method displays a basic description of a car, and then calls ShowDetails to provide additional information.

Each of the three classes defines a ShowDetails method.

The new modifier is used to define ShowDetails in the ConvertibleCar class.

The override modifier is used to define ShowDetails in the Minivan class.

// Define the base class, Car. The class defines two methods,
// DescribeCar and ShowDetails. DescribeCar calls ShowDetails, and each derived
// class also defines a ShowDetails method. The example tests which version of
// ShowDetails is selected, the base class method or the derived class method.
class Car
{
public void DescribeCar()
{
System.Console.WriteLine("Four wheels and an engine.");
ShowDetails();
} public virtual void ShowDetails()
{
System.Console.WriteLine("Standard transportation.");
}
} // Define the derived classes. // Class ConvertibleCar uses the new modifier to acknowledge that ShowDetails
// hides the base class method.
class ConvertibleCar : Car
{
public new void ShowDetails()
{
System.Console.WriteLine("A roof that opens up.");
}
} // Class Minivan uses the override modifier to specify that ShowDetails
// extends the base class method.
class Minivan : Car
{
public override void ShowDetails()
{
System.Console.WriteLine("Carries seven people.");
}
}

The example tests which version of ShowDetails is called.

The following method, TestCars1, declares an instance of each class, and then callsDescribeCar on each instance.

public static void TestCars1()
{
System.Console.WriteLine("\nTestCars1");
System.Console.WriteLine("----------"); Car car1 = new Car();
car1.DescribeCar();
System.Console.WriteLine("----------"); // Notice the output from this test case. The new modifier is
// used in the definition of ShowDetails in the ConvertibleCar
// class. ConvertibleCar car2 = new ConvertibleCar();
car2.DescribeCar();
System.Console.WriteLine("----------"); Minivan car3 = new Minivan();
car3.DescribeCar();
System.Console.WriteLine("----------");
}

TestCars1 produces the following output.

Notice especially the results for car2, which probably are not what you expected.

The type of the object isConvertibleCar, but DescribeCar does not access the version of ShowDetails that is defined in the ConvertibleCar class because that method is declared with the new modifier, not the override modifier.

As a result, a ConvertibleCar object displays the same description as a Car object.

Contrast the results for car3, which is a Minivan object.

In this case, the ShowDetails method that is declared in the Minivan class overrides theShowDetails method that is declared in the Car class, and the description that is displayed describes a minivan.

// TestCars1
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Carries seven people.
// ----------

TestCars2 creates a list of objects that have type Car. The values of the objects are instantiated from the Car, ConvertibleCar, and Minivanclasses. DescribeCar is called on each element of the list. The following code shows the definition of TestCars2.

public static void TestCars2()
{
System.Console.WriteLine("\nTestCars2");
System.Console.WriteLine("----------"); var cars = new List<Car> { new Car(), new ConvertibleCar(),
new Minivan() }; foreach (var car in cars)
{
car.DescribeCar();
System.Console.WriteLine("----------");
}
}

The following output is displayed.

Notice that it is the same as the output that is displayed by TestCars1.

The ShowDetails method of the ConvertibleCar class is not called, regardless of whether the type of the object is ConvertibleCar, as in TestCars1, or Car, as in TestCars2.

Conversely相反地, car3 calls the ShowDetails method from the Minivan class in both cases, whether it has type Minivan or type Car.

// TestCars2
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Carries seven people.
// ----------

Methods TestCars3 and TestCars4 complete the example.

These methods call ShowDetails directly, first from objects declared to have typeConvertibleCar and Minivan (TestCars3),

then from objects declared to have type Car (TestCars4).

The following code defines these two methods.

public static void TestCars3()
{
System.Console.WriteLine("\nTestCars3");
System.Console.WriteLine("----------");
ConvertibleCar car2 = new ConvertibleCar();
Minivan car3 = new Minivan();
car2.ShowDetails();
car3.ShowDetails();
} public static void TestCars4()
{
System.Console.WriteLine("\nTestCars4");
System.Console.WriteLine("----------");
Car car2 = new ConvertibleCar();
Car car3 = new Minivan();
car2.ShowDetails();
car3.ShowDetails();
}

The methods produce the following output, which corresponds to the results from the first example in this topic.

// TestCars3
// ----------
// A roof that opens up.
// Carries seven people. // TestCars4
// ----------
// Standard transportation.
// Carries seven people.

The following code shows the complete project and its output.

Knowing When to Use Override and New Keywords (C# Programming Guide)的更多相关文章

  1. Versioning with the Override and New Keywords (C# Programming Guide)

    The C# language is designed so that versioning between base and derived classes in different librari ...

  2. virtual (C# Reference)

    https://msdn.microsoft.com/en-us/library/9fkccyh4.aspx The virtual keyword is used to modify a metho ...

  3. Polymorphism (C# Programming Guide)

    https://msdn.microsoft.com/en-us/library/ms173152.aspx Polymorphism is often referred to as the thir ...

  4. override (C# Reference)

    https://msdn.microsoft.com/en-us/library/ebca9ah3.aspx The override modifier is required to extend o ...

  5. .NET 面试题: C# override && overloading (C# 覆写 && 重载)

    1 1 1 .NET 面试题, C# ,override , overloading, 覆写, 重载,.NET,ASP.NET, override (覆写/重写): 方法名相同,参数的个数和类型相同, ...

  6. How to Write Doc Comments for the Javadoc Tool

    http://www.oracle.com/technetwork/java/javase/documentation/index-137868.html This document describe ...

  7. Instruments 使用指南

    Instruments 用户指南 http://cdn.cocimg.com/bbs/attachment/Fid_6/6_24457_90eabb4ed5b3863.pdf 原著:Apple Inc ...

  8. Understanding the Objective-C Runtime

    Wednesday, January 20, 2010 Understanding the Objective-C Runtime The Objective-C Runtime is one of ...

  9. SpringBoot+Mybatis配置Pagehelper分页插件实现自动分页

    SpringBoot+Mybatis配置Pagehelper分页插件实现自动分页 **SpringBoot+Mybatis使用Pagehelper分页插件自动分页,非常好用,不用在自己去计算和组装了. ...

随机推荐

  1. PHP--选择排序

    <?php /** * 选择排序(从小到大)的思想:每一次从待排序的数据中选出最小的,放在待排序的起始位置. */ $arr = array(23, 42, 21, 8, 4, 2, 3, 1) ...

  2. Vscode下调试基于Homestead环境的Laravel框架

    PS:最近在学Laravel框架,本机IDE是Vscode,因为Vscode是真的好用!今天突然想调试php代码了,于是疯狂地在网上查资料,经过一上午的不懈努力,终于成功了! 准备工作 首先环境要保证 ...

  3. python_ 学习笔记(hello world)

    python中的循环语句 循环语句均可以尾随一个else语句块,该块再条件为false后执行一次 如果使用break跳出则不执行. for it in [1,2,3,4]: print(it,end= ...

  4. Go:字符串操作

    Package strings:https://golang.google.cn/pkg/strings/ package main import ( "fmt" "st ...

  5. 51nod1128 正整数分组V2

    [题解] 二分一个最大值,check一下分出来的组数是否小于等于k即可. #include<cstdio> #include<algorithm> #define LL lon ...

  6. CTSC2018 Day2T1 Juice混合果汁

    [题解] 在考场上A掉的题. 把美味度排个序,然后按照价格p为权值建立主席树,把每个果汁按照拍好的顺序添加进去.主席树上维护总升数cnt以及总价格sum.对于每个询问,我们二分一个美味值,check的 ...

  7. RabbitMq的简单使用

    本篇将介绍RabbitMq的一个简单使用例子,分别介绍生产者如何发送消息,消费者如何接收和处理消息 关于RabbitMQ的知识背景的文章非常多.我对它的总结是,解决高并发请求的瓶颈,将应用程序真正处理 ...

  8. Leetcode 99.恢复二叉搜索树

    恢复二叉搜索树 二叉搜索树中的两个节点被错误地交换. 请在不改变其结构的情况下,恢复这棵树. 示例 1: 输入: [1,3,null,null,2] 输出: [3,1,null,null,2] 示例  ...

  9. noip模拟赛 天天和不可描述

    分析:直接就这么翻肯定是不行的,换一种想法:有括号就是把括号里的字符串倒着输出,如果在括号里又遇到了括号就继续倒着输出,相当于递归. 我们可以用递归直接做,也可以用一层循环搞定,每次从左括号跳到右括号 ...

  10. 转载 - Struts2拦截器配置

    出处:http://blog.csdn.net/axin66ok/article/details/7321430 目录(?)[-] 理解拦截器 1 什么是拦截器 2 拦截器的实现原理 拦截器的配置 使 ...