简介
面向对象开发有三大特性(特点/特征):封装,继承,多态。
在继承关系的前提下,实例化出不同的对象,这些对象调用相同的方法,但是却表现出不同的行为,这就叫做多态。
在C#语言中体现多态有三种方式:虚方法,抽象类,接口。
虚方法
- 在父类中使用
virtual
关键字修饰的方法,就是虚方法。 - 在子类中可以使用
override
关键字对该虚方法进行重写。
// 父类:
public virtual 返回值类型 方法名(){
方法体代码;
}
// 子类:
public override 返回值类型 方法名(){
方法体代码;
}
- 将父类的方法标记为虚方法,就是在父类方法的返回值前加virtual关键字,表示这个方法可以被子类重写。
- 父类中用virtual修饰的方法,可以用于实现该方法共有的功能(比如初始化该方法),然后在子类重写该方法时,使用base关键字调用父类中的该方法。
class Cat:CatType{
public override void Cry(){
base.Cry();
Console.WriteLine("喵喵喵");
}
}
里氏转换原则
- 子类对象可以直接赋值给父类的变量
- 子类对象可以调用父类中的成员(方法),但是父类对象永远只能调用自己的成员
- 如果父类对象中装的是子类对象,可以将这个父类对象强转为子类对象
Cat c = new Cat();
c.method_in_father(); //调用成功,对应第二条第一句
CatType ct = new Cat(); //赋值成功,对应第一条
ct.Cry(); //调用的是重写后的
c.method_in_father(); //调用成功,对应第二条第二句
c.method_in_son(); //调用失败,同上,即使他是一个子类对象赋值过去的
Cat c2 = (Cat)ct; //转换成功,对应第三句
is
和as
两个关键字都可以进行类型转换。
is
:如果转换成功,返回true,失败返回false;
as
:如果转换成功,返回对应的对象,失败返回null。
bool bl = ct is Tiger; //输出False
bool bl = ct is Cat; //输出True
if (ct as Cat == null){
Console.WriteLine("转换失败");
}
else{
Console.WriteLine("转换成功");
}
//输出转换成功
抽象类
·抽象方法
- 虚方法->抽象方法
父类里面用virtual关键字修饰的方法叫做虚方法,子类可以使用override重新该虚方法,也可以不重写。虚方法还是有方法体的,当我们父类中的这个方法已经虚到完全无法确定方法体的时候,就可以使用另外一种形式来表现,这种形式叫抽象方法。
抽象方法的返回值类型前用关键字abstract修饰,且无方法体。
abstract class NPC{
public abstract void Speak();
}
抽象方法必须存在于抽象类中。
·抽象类
//父类
abstract class NPC{
public abstract void Speak();
}
//子类
class ShopNPC:NPC{
public override void Speak();
}
子类继承抽象类,使用override关键字重写父类中所有的抽象方法。
- 抽象类中不一定要有抽象方法,但是抽象方法必须存在于抽象类中。
- 抽象类不能被实例化,因为抽象类中有抽象方法(无方法体),如果真能实例化抽象类的话,调用这些无方法体的方法是没有任何意义的,所以无法实例化。
·使用场景
<1>当父类中的方法不知道如何去实现的时候,可以考虑将父类写成抽象类,将方法写成抽象方法。
<2>如果父类中的方法有默认实现,并且父类需要被实例化,这时可以考虑将父类定义成一个普通类,用虚方法实现多态。
<3>如果父类中的方法没有默认实现,父类也不需要被实例化,则可以将该类定义为抽象类。
接口
- 接口使用interface关键字定义,没有class关键字,接口名一般使用“IXxxx”
- 接口中不能包含字段,但是可以包含自动属性
- 接口中定义的方法不能有方法体,全是抽象方法,但又不需要用abstract修饰;
- 接口中的成员不允许添加访问修饰符,默认都是public;
interface IFly{
void Fly();
public int Age{ get; set; } //自动属性
}
接口注意事项
- 接口中所有的方法都是抽象方法,所以接口不能被实例化;
- 一个类可以实现多个接口,被实现的多个接口之间用逗号分隔开;
- 一个接口可以继承多个接口,接口之间也要用逗号分隔。
三者比较
使用场景对比
-
虚方法:父类中的个别方法用虚方法实现,然后允许子类在有需要的情况下重写这些虚方法。
-
抽象类:父类定义一系列的规范,子类去把父类里面定义的这些规范全部实现。
-
接口:是一种功能的扩展,是在原有的类的继承关系以外的新功能的扩展。