Virtual :
virtual 关键字用于修饰方法、属性、索引器或事件声明,并使它们可以在派生类中被重写。
默认情况下,方法是非虚拟的。不能重写非虚方法。
virtual 修饰符不能与 static、abstract, private 或 override 修饰符一起使用。
Override :
override 方法提供从基类继承的成员的新实现。
由 override 声明重写的方法称为重写基方法。重写的基方法必须与 override 方法具有相同的签名。
重写的基方法必须是vitural,override,abstract类型的 ,不能重写非虚方法或是静态方法。
所以override不能与vitural,new,static同时使用。
override 属性,声明必须指定与继承属性完全相同的访问修饰符、类型和名称,并且被重写的属性必须是 virtual、abstract 或 override 的。(注意:这里是属性,不是字段,字段是不能被重写的)。
New :
用于创建对象和调用构造函数。
使用 new 修饰符可以显式隐藏从基类继承的成员。
若要隐藏继承的成员,请使用相同名称在派生类中声明该成员,并用 new 修饰符修饰它。
//下面用代码来说明: //By Alan Liao 2008-10-22 class Father { protected string firstname; protected string lastname; public Father() { } public Father(string firstname, string lastname) { this.firstname = firstname; this.lastname = lastname; } public virtual void GetFullName() { Console.WriteLine("Father:{0} {1}", firstname, lastname); } } class Son : Father { public int age; public Son() : base() { } public Son(string firstname, string lastname,int age):base(firstname,lastname) { this.age = age; } //public new void GetFullName() public override void GetFullName() { Console.WriteLine("Son:{0} {1}", firstname, lastname); } } class Program { static void Main(string[] args) { Father father = new Father("Alan", "Liao"); Father son1 = new Son("Little Alan ", "Liao",20); father.GetFullName(); son1.GetFullName(); Console.ReadLine(); } }
本代码输出:
Father:Alan Liao
Son:Little Alan Liao
若将36行换成35行(即将Override换成New)
输出结果为:
(其实代码中的virtual,override 都可以去掉,可以编译,运行通过,但是会有Warnning,结果同下)
Father:Alan Liao
Father:Little Alan Liao
可见结果不一样。
Father a=new Son();
使用new声明的方法使用的仍是父类方法,而override声明的方法使用的是子类中的方法。
其实,根本原因是对象a会优先调用子类中的方法,如果子类没有重写父类的方法,a 就调用父类中的方法。
override 说明的是虚方法,在子类中覆盖父类中的方法。
new则不是,new是给子类新增加了一个函数,可以说除了函数名字相同以外,它跟基类里面的这个同名函数没有任何关系。相当于重新新建了一个方法,从而隐藏了父类方法.
其它:
在 C# 中,派生类可以包含与基类方法同名的方法。
(1)基类方法必须定义为 virtual。
(2)如果派生类中的方法前面没有 new 或 override 关键字,则编译器将发出警告,
该方法将有如存在 new 关键字一样执行操作。
(3)如果派生类中的方法前面带有 new 关键字,则该方法被定义为独立于基类中的方法。
(4)如果派生类中的方法前面带有 override 关键字,则派生类的对象将调用该方法,而不是调用基类方法。
(5)可以从派生类中使用 base 关键字调用基类方法。
(6) override、virtual 和 new 关键字还可以用于属性、索引器和事件中。
补充:
class Father { public virtual void PrintName() { Console.WriteLine("Father:{0}", this.GetType().ToString()); } } class Son : Father { //public new void PrintName() public override void PrintName() { Console.WriteLine("Son:{0}", this.GetType().ToString()); } } class Program { static void Main(string[] args) { Father father = new Father(); Father son = new Son(); father.PrintName(); son.PrintName(); Console.ReadLine(); } }
以上程序的运行结果:(其中myNamespace是这两个类的命名空间)
Father:myNamespace.Father
Son:myNamespace.Son
如果在第11行将override换成new,那么运行结果如下:
Father:myNamespace.Father
Father:myNamespace.Son
从上可以看出虽然调用的是Father类中的方法,但是this.GetType()返回的是myNamespace.Son类型,所以在Father中的PrintName()方法中的this指的是调用该方法的对象,而不是定义该方法的对象。