一.定义
简单的说,虚方法就是可以被子类重写的方法,如果子类重写了虚方法,那么运行时将使用重写后的逻辑,如果没有重写,则使用父类中虚方法的逻辑。
定义关键字:virtual
重写关键字:override
二.实例
定义一个Person类与Person类的Student子类,如下:
public class Person
{
public string Name { get; set; }
public virtual int Age { get; set; }
public virtual void SayHello()
{
Console.WriteLine("大家好,我是"+this.Name+" 年龄"+this.Age);
}
public virtual int Add(int a)
{
return this.Age+a;
}
}
假设现在学生类有如下需求:
1.需要在SayHello方法中增加学号的返回
2.需要对年龄做出校验,当年龄大于18岁时,直接返回18
public class Student:Person
{
public string StuNO { get; set; }
public override int Age
{
get => base.Age;
set => base.Age = value>18?18:value;
}
public override void SayHello()
{
Console.WriteLine("大家好,我是" + this.Name+",年龄"+this.Age+",我的学号是"+this.StuNO);
}
}
我们看到,针对需求1和2,重写了Age属性与SayHello方法,在控制台调用SayHello方法,调用方法如下:
Person person = new Person();
person.Name = "人";
person.Age = 25;
person.SayHello();
Student student = new Student();
student.Name = "学生";
student.Age = 25;//这里的年龄大于18
student.StuNO = "001";
student.SayHello();
结果如下图:
这个时候,我们在Student类中增加一个Add方法(不是重写),然后再重写父类中的Add方法,如下:
public int Add(int a,int b) //这个是新方法,表明覆盖父类里的同名类,而不是重新实现
{
return this.Age + a + 10;
}
public override int Add(int a) //这个是重写的父类中的Add方法
{
return base.Add(a)+666;
}
这时,在控制台调用Add方法时,如果采用两个参数,则执行新方法,如果采用一个参数,则执行重写方法。
如果采用子类实例化父类的方式,如下:
Person aaa = new Student();
aaa.SayHello();
通过验证,我们会发现,这时的SayHello执行的是子类中重写的SayHello方法
关于子类与父类虚方法的实现顺序,在这里直接给出前人的总结:
1、当调用一个对象的函数时,系统会直接去检查这个对象申明定义的类,即申明类,看所调用的函数是否为虚函数;
2、如果不是虚函数,那么它就直接执行该函数。而如果有virtual关键字,也就是一个虚函数,那么这个时候它就不会立刻执行该函数了,而是转去检查对象的实例类。
3、在这个实例类里,他会检查这个实例类的定义中是否有重新实现该虚函数(通过override关键字),如果是有,那么OK,它就不会再找了,而马上执行该实例类中的这个重新实现的函数。而如果没有的话,系统就会不停地往上找实例类的父类,并对父类重复刚才在实例类里的检查,直到找到第一个重载了该虚函数的父类为止,然后执行该父类里重载后的函数。
至此,虚方法的总结结束......