实现继承+接口继承+虚方法+隐藏方法+this/base+抽象类+密封类/方法+修饰符
概念:
在上一节课中学习了如何定义类,用类当做模板来声明我们的数据。
很多类中有相似的数据,比如在一个游戏中,有Boss类,小怪类Enemy,这些类他们有很多相同的属性,也有不同的,这个时候我们可以使用继承来让这两个类继承自同一个类。
分类:
实现继承: 表示一个类型派生于一个基类型,它拥有该基类型的所有成员字段和函数。 在实现继承中,派生类型采用基类型的每个函数的实现代码,除非在派生类型的定义中指定重写某个函数的实现代码。 在需要给现有的类型添加功能,或许多相关的类型共享一组重要的公共功能时,这种类型的继承非常有用。
接口继承: 表示一个类型只继承了函数的签名,没有继承任何实现代码。 在需要指定该类型具有某些可用的特性时,最好使用这种类型的继承。
多重继承:
C#不支持多重实现继承。 而 C#允许类型派生自多个接 口— — 多重接口继承。 这说明,C#类可以派生自另一个类和任意多个接口。更准确地说, System.Object 是一个公共的基类,所 以每个 C#(除了Object类之外)都有一个基类,还可以有任意多个基接 口。
实现继承:
如果要声明派生自另一个类的一个类,就可以使用下面的语法: class MyDerivedClass : MyBaseclass { // functions and data members here } 如果类(或 结构)也 派生 自接 口,则用逗号分隔列表中的基类和接 口: public class MyDerivedClass: MyBaseClass , IInterface1 , IInterface2 { // etc. }
接口继承:
namespace _012_定义和实现接口 { interface IA //不能有字段,方法不能被实现 { void Method1(); } } namespace _012_定义和实现接口 { interface IB:IA { void Method2(); } } namespace _012_定义和实现接口 { interface IFly { void Fly(); void MethodA(); } } //继承了接口就要实现继承的接口中的所有方法,包括接口的父接口中的方法 namespace _012_定义和实现接口 { class Bird :IFly,IB{ public void Fly() { } public void MethodA() { } public void Method1() { } public void Method2() { } } }
虚方法:
把一个基类函数声明为virtual,就可以在任何派生类中重写该函数: class MyBaseClass{ public virtual string VirtualMethod(){ return "Method is called in base class"; } } 在派生类中重写另外一个函数时,要使用override关键字显示声明 class MyDerivedClass:MyBaseClass{ public override string VirtualMethod(){ return "Method is called in derivedclass."; } } 我们在子类里面重写虚函数之后,子类对象不管在哪里调用都是调用重写之后的方法,可以不重写,不重写就调用父类的方法,将子类对象赋给父类对象也会调用子类中重写的方法
隐藏方法(密封方法):
如果签名相同的方法在基类和派生类中都进行了声明,但是该方法没有分别声明为virtual和override,派生类就会隐藏基类方法。(要使用new关键字进行声明) 基类 class MyBaseClass{ public int MyMethod(){ } } 派生类(在派生类中把基类同名的方法隐藏掉了) class MyDerivedClass :MyBaseClass{ public new void MyMethod() //有没有new都可以,只是表示是隐藏方法,但是父类的方法还是存在的,重写的话父类的方法在子类中是不存在了
{ } }
Enemy enemy;
enemy = new Boss();//父类声明的对象,可以使用子类去构造 子类声明的对象不可以使用父类构造
//enemy虽然使用父类进行了声明,但是使用了子类构造,所以本质上是一个子类类型的,我们可以强制类型转换转换成子类类型
Boss boss = (Boss)enemy;
boss.Attack();
Enemy enemy = new Enemy();
Boss boss =(Boss) enemy;//一个对象是什么类型的 主要看它是通过什么构造的 这里enemy使用了父类的构造函数,所以只有父类中的字段和方法, 不能被强制转换成子类
Enemy boss = new Boss();
boss.Move();//隐藏方法: 如果使用子类声明的对象,调用隐藏方法会调用子类的,如果使用父类声明对象,那么就会调用父类中的隐藏方法
this和base关键字:
this可以访问当前类中定义的字段,属性和方法,有没有this都可以访问,有this可以让IDE-VS编译器给出提示,另外当方法的参数跟字段重名的时候,使用this可以表明访问的是类中的字段。
base可以调用父类中的公有方法和字段,有没有base都可以访问,但是加上base.IED工具会给出提示,把所有可以调用的字段和方法罗列出来方便选择
使用base关键字调用父类构造函数的语法如下: 子类构造函数:base(参数列表) 使用base关键字调用父类方法的语法如下: base.父类方法();
using System ; class Teacher//老师类 { public Teacher()//构造函数1 { Console.WriteLine ("我是一位教师。"); } public Teacher(string str)//构造函数2 { Console.WriteLine ("老师,"+str); } public void OutPut()//自定义方法 { Console.WriteLine ("输出方法"); } private string name; public string Name//属性 { get{return this.name;} set{this.name=value;} } public void getName() { Console.WriteLine ("我的名字是"+name); } } class Jack:Teacher { static string hello="你好"; public Jack():base(hello)//子类的构造函数继承的为父类第二个构造函数,注意写法 { } public void myOutPut()//自定义函数 { base.OutPut ();//引用父类的函数 } public string myName//自定义属性 { get{return base.Name ;} set{base.Name ="王"+value;} } } class Test { static void Main() { Jack j=new Jack ();//输出“老师,你好” j.myOutPut ();//输出"输出方法" j.myName ="猫骨"; j.getName ();//输出“王猫骨” } } 注意:base()调用父类构造函数时,不需要再次指定参数的类型,因为在子类中已经定义了这些参数,在base()中只需指定变量名即可,参数的类型必须和父类中的一致。
抽象类:
C#允许把类和函数声明为 abstract。 抽象类不能实例化,抽象类可以包含普通函数和抽象函数,抽象函数就是只有函数定义没有函数体。 显然,抽象函数本身也是虚拟的Virtual(只有函数定义,没有函数体实现)。
类是一个模板,那么抽象类就是一个不完整的模板,我们不能使用不完整的模板去构造对象。,但是可以声明对象
abstract class Bird//一个抽象类 就是一个不完整的模板 { private float speed; public void Eat() { } public abstract void Fly();//抽象方法,没有实体 } class Crow:Bird {//我们继承了一个抽象类的时候,必须去实现抽象方法 public override void Fly() { Console.WriteLine("乌鸦在飞行"); } }
密封类和密封方法:
C#允许把类和方法声明为 sealed。 对于类 ,这表示不能继承该类;对于方法表示不能重写该方法。 sealed FinalClass { // etc } 什么时候使用 密封类和密封方法? 防止重写某些类导致代码混乱,在重写的方法override前加sealed 商业原因
派生类的构造函数:
1,在子类中调用父类的默认构造函数(无参)(会先调用父类的,然后是子类的) public class MyDerivedClass{ public MyDerivedClass():base(){ //do something } } 在这里 :base()可以直接不写,因为默认会调用父类中的默认构造函数 2,调用有参数的构造函数 public class MyDerivedClass{ public MyDerivedClass(string name):base(name){ //do something } }
修饰符:
修饰符,用来类型或者成员的关键字。修饰符可以指定方法的可见性。
public 和private修饰字段和方法的时候,表示该字段或者方法能不能通过对象去访问:
只有public的才可以通过对象访问
private(私有的)只能在类模板内部访问。
protected 保护的,当没有继承的时候,它的作用和private是一样的,当有继承的时候,protected表示可以被子类访问的字段或者方法
类型的修饰符:
public class ...
class ...
前者可以在别的项目下访问,后者不行
其他修饰符:
static可以修饰字段或者方法,修饰字段的时候,表示这个字段是静态的数据,叫做静态字段或者静态属性,修饰方法的时候,叫做静态方法,或者静态函数 使用static修饰的成员,只能通过类名访问 当我们构造对象的时候,对象中只包含了普通的字段,不包含静态字段