继承第四章(上)方法重载应遵循三大原则;适当地重载构造方法,可以使初始化对象的方式更为灵活;this引用总是指向调用成员方法的对象本身;静态成员(包括静态成员属性和静态成员方法)属于类而不是属于某个对象,可以在实例化对象之前通过类名访问;使用package关键字打包;使用import关键字导入包或类。回顾本章相关词汇(蓝色为关键字)单词说明protected受保护的extends扩充,扩展super上一级的,超级的access访问inheritance继承,遗传baseclass基类/父类derivedclass子类/派生类modifier修饰符本章目标对象数组(补充)理解什么是继承在Java中实现继承,extends关键字4种访问控制权限publicprotectedprivate缺省继承中的构造方法super关键字对象数组在Java中不但可以声明由原始数据组成的数组,还可以声明由对象组成的数组;声明对象数组的方式如:但上述方式类似于C语言中的:structStudent**p=NULL;p=(structStudent**)malloc(5*sizeof(structStudent*));//只是声明了一个指针数组,没有任何实际的结构体变量//假设已经定义了一个Student类/*声明由5个Student对象组成的数组*/Student[]stdAry=newStudent[5];对象数组==引用数组错误的理解方式学生对象的实例学生对象的实例学生对象的实例学生对象的实例学生对象的实例01234堆栈stdAry堆栈stdAry正确的理解方式对象的引用01234对象的引用对象的引用对象的引用对象的引用对象数组示例/*对象数组示例,假设已经定义了Student类*/publicstaticvoidmain(String[]args){/*创建包含有5个Student引用的数组*/Student[]stdAry=newStudent[5];/*逐一为每个引用创建对象实例*/stdAry[0]=newStudent(张三,18);stdAry[1]=newStudent(李四,20);stdAry[2]=newStudent(王五,24);stdAry[3]=newStudent(郑六,21);stdAry[4]=newStudent(田七,19);for(inti=0;istdAry.length;i++){stdAry[i].display();}}对象数组的存放形式堆栈stdAry0123stdAry[0]stdAry[1]stdAry[2]stdAry[3]stdAry[4]4张三18李四20王五24郑六21田七19面向对象的三大特征面向对象的程序设计有三大特征:封装:解决了数据的安全性问题继承:解决了代码的重用问题多态:解决了程序的扩展问题前面的章节我们已经学习了有关于封装的各个概念,这一章我们来讨论第二大特征——继承。继承的概念在现实生活中的继承,可以理解为儿子继承了父亲的财产,即财产重用;面向对象程序设计中的继承,则是代码重用;继承是利用现有的类创建新类的过程,现有的类称作基类(或父类),创建的新类称作派生类(子类)。基类方法和属性派生类基类方法+附加方法大学系统人员分类树人学生老师大学生研究生继承的概念(续)最高层是最普遍的、最一般的情况,往下每一层都比上一层更具体,并包含有高层的特征,通过这样的层次结构使下层的类能自动享用上层类的特点和性质;继承其实就是自动地共享基类中成员属性和成员方法的机制。在Java中实现继承在Java中实现继承需要使用到extends关键字;实现继承的一般语法是:[访问修饰符]class派生类名extends基类名{成员列表}如:classStudentextendsPerson{……}实现继承示例classPerson{//定义人类publicStringmName;//姓名publicintmAge;//年龄publicvoiddining(){System.out.println(吃饱了...);}//吃饭的方法}classStudentextendsPerson{//学生类继承于人类publicfloatmGrade;//成绩publicvoidexamination(){System.out.println(考试及格了...);}//考试的方法}classTeacherextendsPerson{//教师类继承于人类publicfloatmSalary;//薪水publicvoidprelection(){System.out.println(上课很累...);}//上课的方法}publicclassInheritanceDemo{//该类用于容纳main方法publicstaticvoidmain(String[]args){Studentstd=newStudent();//实例化学生对象std.mName=张三;std.mAge=18;//为姓名和年龄赋值,访问的是父类中的成员std.dining();//调用吃饭的方法,访问的是父类中的成员std.examination();//调用考试方法,访问的是子类中的成员Teachertea=newTeacher();//实例化教师对象tea.mName=谭浩强;tea.mAge=65;tea.dining();tea.prelection();}}继承的作用当今软件设计的特征:软件规模越来越大;软件设计者越来越多;软件设计分工越来越细。引入继承,实现了代码重用;引入继承,实现了递增式的程序设计。继承的作用(续)继承是能自动传播代码和重用代码的有力工具;继承能够在某些比较一般的类的基础上建造、建立和扩充新类;能减少代码和数据的重复冗余度,并通过增强一致性来减少模块间的接口和界面,从而增强了程序的可维护性;能清晰地体现出类与类之间的层次结构关系。与继承有关的注意事项继承是单方向的,即派生类可以继承和访问基类中的成员,但基类则无法访问派生类中的成员;在Java中只允许单一继承方式,即一个派生类只能继承于一个基类,而不能象C++中派生类继承于多个基类的多重继承方式。类成员的访问控制权限信息隐藏是面向对象程序设计的重要特点之一,它可以:防止类的使用者意外损坏数据;对任何实现细节所作的修改不会影响到使用该类的其它代码;使类更易于使用。在Java中实现信息隐藏的是访问控制权限机制;访问控制权限包括4个访问修饰符:public、protected、private和缺省;可以使用上述访问修饰符修饰类的成员。访问修饰符不受任何限制,本类或非本类均可随意访问。public本类及其子类可以访问(父子友好),同一个包中的其它类也可访问(包内友好)。protected只有相同包中的类可以访问(包内友好)。缺省只有本类可以访问,其余都不可以。private访问控制权限(列表)publicprotected缺省private本类可以可以可以可以同包子类可以可以可以不可以同包非子类可以可以可以不可以不同包子类可以可以不可以不可以不同包且非子类可以不可以不可以不可以访问修饰符位置课堂练习1源文件BaseClass.javapackagemypkg;publicclassBaseClass{publicintpubA;protectedintproB;intdefC;privateintpriD;}源文件DerivedClass.javapackagemypkg;publicclassDerivedClassextendsBaseClass{publicvoidfun(){pubA=10;proB=20;defC=30;priD=40;}}√√√×课堂练习2源文件Frist.javapublicclassFrist{publicintpubA;protectedintproB;intdefC;privateintpriD;}publicclassSecond{publicvoidfun(){Fristobj;obj=newFrist();obj.pubA=10;obj.proB=20;obj.defC=30;obj.priD=40;}}源文件Second.java√√×√课堂练习3源文件SuperClass.javapackageaaa;publicclassSuperClass{publicintpubA;protectedintproB;intdefC;privateintpriD;}源文件SubClass.javapackagebbb;importaaa.SuperClass;publicclassSubClassextendsSuperClass{publicvoidfun(){pubA=10;proB=20;defC=30;priD=40;}}√√××类的访问权限还可以在定义类时为类添加访问修饰符,对类进行访问权限控制;对类使用的访问修饰符只有public和缺省两种;被public修饰的类可以从任何地方访问,不受限制;不加访问修饰符,缺省修饰的类只能从本包中访问,不同包则无法访问到;但要注意的是:在一个源文件中只能有一个被public修饰的类,并且文件名必须与public的类同名;如果要定义多个public的类,则必须分别写在不同的源文件中,一个源文件只写一个类是良好的编程习惯。继承中的构造方法父类中的构造方法不能被子类继承,即便它是public的;父类的构造方法负责初始化属于它的成员变量,而子类的构造方法则只需考虑属于自己的成员变量,不必去关注父类的情况。继承中的构造方法示例classParentClass{//定义父类publicParentClass(){//构造方法System.out.println(这是父类的构造方法。);}}classChildClassextendsParentClass{//子类继承于父类publicChildClass(){//构造方法System.out.println(这是子类的构造方法。);}}publicclassConstructorTest{//该类用于容纳main方法publicstaticvoidmain(String[]args){ChildClasscc=newChildClass();//实例化子类对象}}构造方法的执行顺序当实例化子类的对象时,必须先执行父类的构造方法,然后再执行子类的构造方法;如果父类还有更上级的父类,就会先调用最高父类的构造方法,再逐个依次地将所有继承关系的父类构造方法全部执行;如果父类的构造方法执行失败,那么子类的对象也将无法实例化。案例classPoint{//定义点类//x轴坐标和y轴坐标,由于准备用于继承,故修饰为protectedprotectedfloatmX,mY;publicPoint(floatx,floaty){//构造方法mX=x;mY=y;}}classCircleextendsPoint{//定义圆类继承于点类protectedfloatmRadius;//半径publicCircle(floatr){//构造方法mRadius=r;}}publicclassDemo{publicstaticvoidmain(String[]args){Circlec=newCircle(2.5f);//实例化圆类对象}}本例将报出错误案例分析在实例化Circle类对象时,虚拟机一定会先调用其父类(Point类)的构造方法;Point类的构造方法需要两个参数来初始化其成员,但此时并没有获得这两个参数,造成Point类的构造方法无法执行;父类的构造方法执行失败从而导致子类(Circle类)的对象也无法创建;问题的关键是:在实例化子类对象时,如何将参数传递给父类的构造方法?这将使用到super关键字。super关键字的第一种