先放出来两个基类和派生类:
public class BaseClass { public virtual void Method1(string desc) { Console.WriteLine(desc + ": Base - Method1"); } public void Method2(string desc) { Console.WriteLine(desc + ": Base - Method2"); } } public class DerivedClass : BaseClass { public override void Method1(string desc) { Console.WriteLine(desc + ": Derived - Method1"); } public new void Method2(string desc) { Console.WriteLine(desc + ": Derived - Method2"); } }
new除了可以创建对象和作为泛型约束之外,还可以作为方法的修饰符,这里讨论的是作为方法的修饰符时与override的异同:
- new和override作用于相同的方法签名(方法名相同,参数个数、顺序、类型相同)
- new用于隐藏父级方法。当派生类从一个基类继承并有一个相同签名的方法时,派生类的方法会显示有一个编译警告,提示要隐藏基类方法的话那就用new来修饰。就是说用new之后派生类这方法是派生类的而已,与基类无关了。
- override用于重写基类方法。当派生类从一个基类继承并有一个相同签名的方法时,如果基类方法是可重写的(方法被virtual、abstract或override修饰),则基类的方法将被派生类重写(覆盖)。就是说用override之后派生类的方法派生类自己重新实现了,基类的方法也被派生类的替代了。
- new和override这两个关键字是互斥的。不可以同时使用。
两者该怎么用呢?且看上面两个父子类。DerivedClass继承了BaseClass,Method1重写了基类的,Method2隐藏了基类的。
单纯地var bc = new BaseClass();或var dc = new DerivedClass();调用的是各自的方法,这点不用讨论了。当BaseClass bcdc = new DerivedClass();时,这里发生了一次向上转型,bcdc从DerivedClass转型为其上级(基类)BaseClass类型;实例bcdc.Method1调用的将是DerivedClass的方法,因为在实例化的时候Method1已经被重写(override)过了;而bcdc.Method2调用的将是BaseClass的方法,因为Method2没有被动到,依然是BaseClass原来的方法。
总结起来就是:在派生类中,相同签名的方法基类的也要重写(影响到)的话就是用override,只想派生类实现给自己的话(不影响基类)就用new。这样实例向上转型后override的会被保留(因为已经被派生类重写),new的会使用基类原来的方法。