面向对象的程序设计-Java张白一第三版第5章.ppt
第5章 消息、继承与多态,5.1 消息 5.2 访问控制 5.3 多态机制 5.4 继承机制 5.5 抽象类、接口与包,5.1 消 息在面向对象技术中,对象与对象之间并不是彼此孤立的,它们之间存在着联系,对象之间的联系是通过消息来传递的。在面向对象的程序中,消息就是数据成员及成员方法的引用。5.1.1 消息的概念在日常生活中,人与人之间要进行交流。某人可以向别人提供服务,例如,他可以开汽车、教学生学习等;同时他也需要别人为他提供服务,例如,他要吃饭但不可能自己去种地,要穿衣不可能自己去织布,他必须请求别人的帮助;同样,他什么时间讲课,也必须得到他人的请求或命令。“请求”或“命令”便是人与人进行交流的手段。,在面向对象的系统中,把“请求”或“命令”抽象成“消息”,对象之间的联系是通过消息传递来实现的。当系统中的其他对象请求这个对象执行某个服务时,它就响应这个请求,完成指定的服务。通常,我们把发送消息的对象称为发送者,把接收消息的对象称为接收者。对象间的联系,只能通过消息传递来进行。对象也只有在收到消息时才被激活,去完成消息要求的功能。,消息就是向对象发出服务请求,是对数据成员和成员方法的引用。因此,它应该含有这些信息:提供服务的对象标识对象名、服务标识方法名、输入信息实际参数、回答信息返回值或操作结果。消息具有三个性质:(1)同一对象可接收不同形式的多个消息,产生不同的响应。(2)相同形式的消息可以发送给不同的对象,各对象所做出的响应可以是截然不同的。(3)消息的发送可以不考虑具体的接收者,对象可以响应消息,也可以对消息不予理会,对消息的响应并不是必须的。,5.1.2 公有消息和私有消息在面向对象系统中,消息分为两类:公有消息和私有消息。当有一批消息同属于一个对象时,由外界对象直接发送给这个对象的消息称为公有消息;对象发送给自己的消息称为私有消息。私有消息对外是不开放的,外界不必了解它。外界对象只能向此对象发送公有消息,而不能发送私有消息,私有消息是由对象自身发送的。,5.1.3 特定于对象的消息特定于对象的消息是指将所有能支持此对象可接收消息的方法集中在一起,形成一个大消息,称为特定于对象的消息。这些消息让对象执行这个方法而不管它可能做什么及怎么做。特定于对象的消息可分为以下三种类型:(1)可以返回对象内部状态的消息。(2)可以改变对象内部状态的消息。(3)可以做一些特定操作,改变系统状态的消息。,【示例程序C5_1.java】不同类型消息的传递示例。class Student public String name;public char sex;public int no;public int age;Student(int cno,String cname,char csex,int cage)name=cname;sex=csex;no=cno;age=cage;,public void showNo()System.out.println(No:+no);public void showName()System.out.println(Name:+name);public void showSex()System.out.println(Sex:+sex);public void showAge()System.out.println(age:+age);class StudentScore private int no;private double score;public void sendScore(int cno,double cscore),/下面两条语句是对象发送给自身的消息,要求给自己的数据成员赋值,/这是一种私有消息,外界是不知道的 no=cno;score=cscore;void printScore()System.out.println(No:+no+score:+score);public class C5_1 public static void main(String args),int m;/下面两句发送new消息给类Student,要求创建st1、st2的对象 Student st1=new Student(101,zhang li,F,18);Student st2=new Student(102,hong bing,M,17);/发送new消息给类StudentScore,要求创建sc1、sc2的对象 StudentScore sc1=new StudentScore();StudentScore sc2=new StudentScore();/*向st1的对象发送显示学号、名字、年龄的消息。这些消息都是公有消息。它们形成 了同一对象可接收不同形式的多个消息,产生不同的响应*/,st1.showNo();/这是一条消息,消息响应的结果是显示st1的对象的学号 st1.showName();/显示对象姓名的消息 st1.showAge();/显示对象年龄的消息 st1.age=20;/修改对象的数据成员的消息,修改st1对象的年龄 m=st1.age;/返回对象的数据成员的消息,将返回消息赋给变量m System.out.println(m=+m);/*向st2的对象发送两个显示信息的消息,与st1的对象相同,显示学号及名字。这些消 息都是公有消息,说明了相同形式的消息可以送给不同的对象,各对象所做出的响应 可以是截然不同的*/,st2.showNo();st2.showName();/向sc1、sc2的对象各发送一个按学号输入成绩单的消息,这些消息都是公有消息 sc1.sendScore(101,97);sc2.sendScore(102,84);/向sc1、sc2的对象各发送一个打印消息,这些消息都是公有消息 sc1.printScore();sc2.printScore();,该程序的运行结果如下:No:101Name:zhang liage:18m=20No:102Name:hong bingNo:101 score:97.0No:102 score:84.0,5.2 访 问 控 制一个类总能够访问自己的数据成员和成员方法。但是,其他类是否能访问这个类的数据成员或成员方法,是由该类的访问控制符及该类数据成员和成员方法的访问控制符决定的。这就是说,访问控制符是一组限定类、数据成员或成员方法是否可以被其他类访问的修饰符。类的访问控制符只有public一个,缺省访问控制符时具有“友好访问”的特性。数据成员和成员方法的访问控制符有public、private、protected和缺省访问控制符等几种。类、数据成员和成员方法的访问控制符及其作用见表5.1。,表5.1 类、数据成员和成员方法的访问控制符及其作用,通过声明类的访问控制符可以使整个程序结构清晰、严谨,减少可能产生的类间干扰和错误。,5.2.1 公共访问控制符publicJava的类是通过包的概念来组织的,简单地说,定义在同一个程序文件中的所有类都属于同一个包。处于同一个包中的类都是可见的,即可以不需任何说明而方便地互相访问和引用。而对于不同包中的类,一般说来,它们相互之间是不可见的,当然也不可能互相引用。然而,当一个类被声明为public时,只要在其他包的程序中使用import语句引入这个public类,就可以访问和引用这个类,并创建这个类的对象,访问这个类内部可见的数据成员和引用它的可见的方法。例如,Java类库中的许多类都是公共类,我们在程序中就是通过import语句将其引入的。,当一个类的访问控制符为public时,表明这个类作为整体对其他类是可见和可使用的,这个类就具有了被其他包中的类访问的可能性。但是,处于不同包中的public类作为整体对其他类是可见的,并不代表该类的所有数据成员和成员方法也同时对其他类是可见的,这得由这些数据成员和成员方法的修饰符来决定。只有当public类的数据成员和成员方法的访问控制符也被声明为public时,这个类的所有用public修饰的数据成员和成员方法才同时对其他类也是可见的。在程序设计时,如果希望某个类能作为公共工具供其他的类和程序使用,则应该把类本身和类内的方法都声明为public。例如,把Java类库中的标准数学函数类math和标准数学函数方法都声明为public,以供其他的类和程序使用。,需要注意的是,数据成员和成员方法的访问控制符被声明为public时,会造成安全性和封装性下降,所以一般应尽量少用。【示例程序A.java和B.java】有无public访问控制符的不同。(1)A.java程序:package c51;public class A double x1;/友好访问的数据成员 public double x2;/公共的数据成员 public double ar(double x)/公共的成员方法 double s;s=x;return s;,(2)B.java程序:package c52;import c51.A;public class B public static void main(String args)double s1;A p1=new A();/创建A类p1的对象/p1.x1=7;x1不是公共数据成员,不能访问 p1.x2=5.2;/访问A类p1的对象的数据成员 s1=p1.ar(8);/访问A类p1的对象的成员方法 System.out.println(“p1.x2=”+p1.x2+“s1=”+s1);,上述两个程序文件的位置及其关系如图5.1左上窗口所示,其运行结果见图5.1右下窗口所示。程序说明:为了说明有无public访问控制符的不同,我们在c51包的A.java文件中创建了A类,并定义了一个没有public访问控制符的数据成员x1和一个由public修饰的数据成员x2;然后,我们又在c52包的B.java文件中创建B类,在B类中创建A类p1的对象实例,并访问该对象的数据成员及成员方法。可以看出,在B类中对于没有public访问控制符的A类p1的对象的数据成员x1是不能访问的,编译时会报错,所以,程序中将其注释掉了。,图5.1 程序A.java和B.java的位置图及运行结果,5.2.2 缺省访问控制符如果一个类没有访问控制符,说明它具有缺省的访问控制特性,这种缺省的访问控制特性称为“友好访问”。友好访问规定只有在同一个包中的对象才能访问和引用这些类,因此,友好访问又称为包访问性。同样道理,类内的数据成员和成员方法如果没有访问控制符来限定,那么它们同样具有“友好访问”的特性,它们也具有包访问性,可以被同一个包中的其他类所访问和引用。,【示例程序C5_2.java和ClassArea.java】缺省访问控制符的程序示例。在ch5包中创建两个类:ClassArea类和C5_2类。其中,ClassArea是缺省访问控制符的类,其功能是计算矩形面积;C5_2是一个公共类,在这个类中创建ClassArea类的实例对象ss,并访问该对象的成员方法。,(1)ClassArea.java文件:package ch5;class ClassArea double lon,wid;/数据成员的修饰符为缺省 double area(double x,double y)/成员方法的修饰符为缺省 double s;/方法内的变量 lon=x;wid=y;s=lon*wid;/求矩形面积 return s;/返回面积值,(2)C5_2.java文件:package ch5;public class C5_2 public static void main(String args)double a=2.2,b=3.1,z;/*在类C5_2中创建被访问ClassArea类的ss的对象,并访问对象的成员方法。这就是说,包中类是可见的,可以互相引用*/ClassArea ss=new ClassArea();z=ss.area(a,b);/访问ss的对象的成员方法 System.out.println(“z=”+z);,上述两个程序文件的位置及其关系如图5.2左上窗口所示,其运行结果见图5.2右下窗口,是z=6.820000000000001。,图5.2 程序C5_2.java和ClassArea.java位置图及运行结果,5.2.3 私有访问控制符private用private修饰的数据成员或成员方法只能被该类自身访问和修改,而不能被任何其他类(包括该类的子类)访问和引用。它提供了最高的保护级别。当其他类希望获取或修改私有成员时,需要借助于类的方法来实现。【示例程序C5_3.java】同一个包中,用private修饰的父类的数据成员不能被子类的实例对象引用,故程序中将该语句变成了注释语句。,class P1 private int n=9;/私有数据成员n int nn;P1()/构造方法 nn=n+;/可以被该类的对象自身访问和修改 void ma()System.out.println(n=+n);/可以被该类的对象自身访问public class C5_3 extends P1/类class C5_3是类P1的子类 public static void main(String args),P1 m1=new P1();System.out.println(“m1.nn=”+m1.nn);/System.out.println(“m1.n=”+m1.n);错,不能引用父类的私有成员 m1.ma();/可以引用P1类自身的成员方法 该程序的运行结果如下:m1.nn=9 n=10,5.2.4 保护访问控制符protected用protected修饰的成员变量可以被三种类引用:该类自身,与它在同一个包中的其他类以及在其他包中的该类的子类。使用protected修饰符的主要作用是允许其他包中的它的子类来访问父类的特定属性。【示例程序C41.java和C42.java】当父类的数据成员用protected修饰时,其他包的子类引用该数据成员的情况。具体情况见程序中的注释。,(1)C41.java程序:package c51;public class C41 protected double x1;double x2;protected double ar(double x)double s;s=x;return s;,(2)C42.java程序:package c52;import c51.C41;public class C42 extends C41 public static void main(String args)double s1;C42 p1=new C42();p1.x1=7;/可以访问,属性x1是protected修饰的 s1=p1.ar(4);/可以访问,方法ar()是protected修饰的,/p1.x2=9;错,不能访问,x2没有访问控制符,不能被另一包的实例对象访问 System.out.println(“p1.x1=”+p1.x1+“s1=”+s1);程序C41.java和C42.java的位置及其关系见图5.3左上窗口所示,其运行结果见图5.3右下窗口,为p1.x1=7.0 s1=4.0。,图5.3 程序C41.java和C42.java位置图及运行结果,5.3 多 态 机 制多态是面向对象系统中的又一重要特性,它描述的是同名方法可以根据发送消息的对象传送参数的不同,采取不同的行为方式的特性。面向对象系统中采用多态,大大提高了程序的抽象程度和简洁性,更重要的是,它最大限度地降低了类和程序模块之间的耦合性,提高了类模块的封闭性,使得它们不需了解对方的具体细节,就可以很好地共同工作。这一点对程序的设计、开发和维护都有很大的好处。,5.3.1 多态的概念多态是指一个程序中具有相同名字而内容不同的方法共存的情况。这些方法同名的原因是它们的最终功能和目的都相同,但是由于在完成同一功能时可能遇到不同的具体情况,因此需要定义包含不同具体内容的方法,来代表多种具体实现形式。多态是面向对象程序设计中的一个重要特性,其目的是提高程序的抽象度、封闭性和简洁性,统一一个或多个相关类对外的接口。Java中提供了两种多态机制:重载与覆盖。,5.3.2 重载当在同一类中定义了多个同名而不同内容的成员方法时,我们称这些方法是重载(override)的方法。重载的方法主要通过形式参数列表中参数的个数、参数的数据类型和参数的顺序等方面的不同来区分。在编译期间,Java编译器要检查每个方法所用的参数数目和类型,然后调用正确的方法。,【示例程序C5_5.java】加法运算重载的例子。import;import;public class C5_5 extends Applet int add(int a,int b)/重载的方法1 return(a+b);double add(double x,double y)/重载的方法2 return(x+y);double add(double x,double y,double z)/重载的方法3 return(x+y+z);,public void paint(Graphics g)g.drawString(“Sum is:”+add(8.5,2.3),5,10);g.drawString(“Sum is:”+add(21,38),5,30);g.drawString(“Sum is:”+add(8.5,2.3,8.5+2.3),5,50);该程序的运行结果如下:Sum is:10.8Sum is:59Sum is:21.6,该类中定义了三个名为add的方法:第一个方法是计算两个整数的和;第二个方法是计算两个浮点数的和;第三个方法是计算三个浮点数的和。编译器根据方法被引用时提供的实际参数,选择并执行对应的重载方法。,5.3.3 覆盖由于面向对象系统中的继承机制,子类可以继承父类的方法。但是,子类的某些特征可能与从父类中继承来的特征有所不同,为了体现子类的这种个性,Java允许子类对父类的同名方法重新进行定义,即在子类中可以定义与父类中已定义的方法同名而内容不同的方法。这种多态被称为覆盖(overload)。由于覆盖的同名方法存在于子类对父类的关系中,因此只需在方法引用时指明引用的是父类的方法还是子类的方法,就可以很容易地把它们区分开来。具体应用请参阅节。,5.4 继 承 机 制继承是面向对象程序设计的又一种重要手段。在面向对象的程序设计中,采用继承的机制可以有效地组织程序的结构,设计系统中的类,明确类间关系,充分利用已有的类来完成更复杂、深入的开发,大大提高程序开发的效率,降低系统维护的工作量。,5.4.1 继承的概念同类事物具有共同性,在同类事物中,每个事物又具有其特殊性。运用抽象的原则舍弃对象的特殊性,抽取其共同性,则可得到一个适应于一批对象的类,这便是一般类;而那些具有特殊性的类称为特殊类。也就是说,如果类B具有类A的全部属性和方法,而且又具有自己特有的某些属性和方法,则把类A称做一般类,把类B叫做类A的特殊类。例如:考虑轮船和客轮这两个类。轮船具有吨位、时速、吃水线等属性,并具有行驶、停泊等服务;客轮具有轮船的全部属性与服务,又有自己的特殊属性(如载客量)和服务(如供餐等)。若把轮船看做一般类,则客轮是轮船的特殊类。,在面向对象程序设计中运用继承原则,就是在每个由一般类和特殊类形成的一般特殊结构中,把一般类的对象实例和所有特殊类的对象实例都共同具有的属性和操作一次性地在一般类中进行显式定义,在特殊类中不再重复地定义一般类中已经定义的东西,但是在语义上,特殊类却自动地、隐含地拥有它的一般类(以及所有更上层的一般类)中定义的属性和操作。特殊类的对象拥有其一般类的全部或部分属性与方法,称做特殊类对一般类的继承。,继承所表达的就是一种对象类之间的相交关系,它使得某类对象可以继承另外一类对象的数据成员和成员方法。若类B继承类A,则属于类B的对象便具有类A的全部或部分性质(数据属性)和功能(操作),我们称被继承的类A为基类、父类或超类,而称继承类B为A的派生类或子类。父类与子类的层次关系如图5.4所示。,图5.4 父类与子类的层次关系,继承避免了对一般类和特殊类之间共同特征进行的重复描述。同时,通过继承可以清晰地表达每一项共同特征所适应的概念范围在一般类中定义的属性和操作适用于这个类本身以及它以下的每一层特殊类的全部对象。运用继承原则使得系统模型更简洁、清晰。,5.4.2 继承的特征一般来说,继承具有下述特征:(1)继承关系是传递的。若类C继承类B,类B继承类A,则类C既有从类B那里继承下来的属性与方法,也有从类A那里继承下来的属性与方法,还可以有自己新定义的属性和方法。继承来的属性和方法尽管是隐式的,但仍是类C的属性和方法。继承是在一些比较一般的类的基础上构造、建立和扩充新类的最有效的手段。(2)继承简化了人们对事物的认识和描述,能清晰地体现相关类间的层次结构关系。,(3)继承提供了软件复用功能。若类B继承类A,那么建立类B时只需要再描述与基类(类A)不同的少量特征(数据成员和成员方法)即可。这种做法能减小代码和数据的冗余度,大大增加程序的重用性。(4)继承通过增强一致性来减少模块间的接口和界面,大大增加了程序的易维护性。,(5)提供多重继承机制。从理论上来说,一个类可以是多个一般类的特殊类,它可以从多个一般类中继承属性与方法,这便是多重继承。Java出于安全性和可靠性的考虑,仅支持单重继承,而通过使用接口机制来实现多重继承。图5.5所示为一个单重继承与多重继承的例子。在这个模型中,“本科生”、“研究生”、“脱产研究生”都为单继承,而“在职研究生”为多重继承,因为它不仅继承“学生”/“研究生”的属性和行为,还继承“教师”的属性和行为。,图5.5 单重继承与多重继承的例子,5.4.3 Java用extends指明继承关系在Java程序设计中,继承是通过extends关键字来实现的。在定义类时使用extends关键字指明新定义类的父类,新定义的类称为指定父类的子类,这样就在两个类之间建立了继承关系。这个新定义的子类可以从父类那里继承所有非private的属性和方法作为自己的成员。实际上,在定义一个类而不给出extends关键字及父类名时,默认这个类是系统类object的子类。下面分不同情况来讲解继承关系。,1数据成员的继承子类可以继承父类的所有非私有的数据成员。【示例程序C5_6.java】数据成员的继承。class A1 int x=25;private int z;/不能被子类继承的私有数据成员zclass C5_6 extends A1/A1是C5_6的父类,C5_6是A1的子类 public static void main(String argS),C5_6 p=new C5_6();System.out.println(“p.x=”+p.x);/输出继承来的数据成员的值/System.out.println(“p.z=”+p.z);错,不能继承private修饰的z 该程序的运行结果如下:p.x=25,2数据成员的隐藏数据成员的隐藏是指在子类中重新定义一个与父类中已定义的数据成员名完全相同的数据成员,即子类拥有了两个相同名字的数据成员,一个是继承自父类的,另一个是自己定义的。当子类引用这个同名的数据成员时,默认操作是引用它自己定义的数据成员,而把从父类那里继承来的数据成员“隐藏”起来。当子类要引用继承自父类的同名数据成员时,可使用关键字super引导,这部分内容将在节介绍。,【示例程序C5_7.java】数据成员的隐藏。class A11 int x=8;/父类中定义了数据成员xclass C5_7 extends A11 int x=24;/子类中也定义了数据成员x public static void main(String argS)int s1,s2;A11 p=new A11();/创建父类p的对象 C5_7 p1=new C5_7();/创建子类p1的对象 s1=p.x;,s2=p1.x;/子类对象引用自己的数据成员,把父类数据成员“隐藏”起来 System.out.println(“s1=”+s1);System.out.println(“s2=”+s2);该程序的运行结果如下:s1=8 s2=24,3成员方法的继承子类可以继承父类的非私有成员方法。下面的程序说明这一问题。【示例程序C5_8.java】成员方法的继承。class A2 int x=0,y=1;void Myp()System.out.println(“x=”+x+“y=”+y);private void Printme()System.out.println(“x=”+x+“y=”+y);,public class C5_8 extends A2 public static void main(String arg)int z=3;C5_8 p1=new C5_8();p1.Myp();/p1.Printme();错,不能继承父类的private方法 该程序的运行结果如下:x=0 y=1,4成员方法的覆盖子类可以重新定义与父类同名的成员方法,实现对父类方法的覆盖(overload)。方法的覆盖与数据成员的隐藏的不同之处在于:子类隐藏父类的数据成员只是使之不可见,父类同名的数据成员在子类对象中仍然占有自己独立的内存空间;子类方法对父类同名方法的覆盖将清除父类方法占用的内存,从而使父类方法在子类对象中不复存在。,【示例程序C5_9.java】成员方法的覆盖。class A3 int x=10;int y=31;public void Printme()System.out.println(“x=”+x+“y=”+y);public class C5_9 extends A3 int z=35;public void Printme()/子类中定义了与父类同名的成员方法,实现覆盖(z=+z);,public static void main(String arg)A3 p2=new A3();/创建父类p2的对象 C5_9 p1=new C5_9();/创建子类p1的对象 p1.Printme();/子类对象引用子类方法,覆盖了父类的同名方法 p2.Printme();/父类对象引用父类方法 该程序的运行结果如下:z=35 x=10 y=31,方法的覆盖中需要注意的是:子类在重新定义父类已有的方法时,应保持与父类完全相同的方法名、返回值类型和参数列表,否则就不是方法的覆盖,而是子类定义自己特有的方法,与父类的方法无关。,5.4.4 this与super1this的使用场合在一些容易混淆的场合,例如,当成员方法的形参名与数据成员名相同,或者成员方法的局部变量名与数据成员名相同时,在方法内借助this来明确表示引用的是类的数据成员,而不是形参或局部变量,从而提高程序的可读性。简单地说,this代表了当前对象的一个引用,可将其理解为对象的另一个名字,通过这个名字可以顺利地访问对象,修改对象的数据成员,调用对象的方法。归纳起来,this的使用场合有下述三种:,(1)用来访问当前对象的数据成员,其使用形式如下:this.数据成员(2)用来访问当前对象的成员方法,其使用形式如下:this.成员方法(参数)(3)当有重载的构造方法时,用来引用同类的其他构造方法,其使用形式如下:this(参数)下面通过例子来说明前两种用法,关于第三种用法,将在节介绍。,【示例程序C5_10.java】this的使用。class A4 int x=0;int y=1;public void Printme()System.out.println(“x=”+x+“y=”+y);System.out.println(“I am an”+this.getClass().getName();/用this来访问当前对象的成员方法,通过this表示当前对象,来打印当前对象的类名。/其中的getClass()和getName()是系统类库中提供的方法,public class C5_10 extends A4 public static void main(String arg)C5_10 p1=new C5_10();p1.Printme();该程序的运行结果如下:x=0 y=1I am an C5_10,【示例程序C5_11.java】this的使用。class AlassArea double x,y;double area(double x,double y)double s;this.x=x;/借助this来表示引用的是类数据成员 this.y=y;s=this.x*this.y;return s;,public class C5_11 extends AlassArea public static void main(String args)double a=2.2,b=3.1,z;C5_11 ss=new C5_11();/创建ss的对象 z=ss.area(a,b);/引用父类对象的成员方法求面积 System.out.println(“z=”+z);该程序的运行结果如下:z=6.820000000000001,【示例程序C5_12.java】计算圆的面积和周长。public class C5_12 public static void main(String args)double x;Circle cir=new Circle(5.0);x=cir.area();(“圆的面积=”+x);x=cir.perimeter();(“圆的周长=”+x);,class Circle double r;/定义半径 定义圆周率 public Circle(double r)/类的构造方法 this.r=r;/通过构造方法给r赋值 double area()/计算圆面积的方法 return PI*r*r;double perimeter()/计算圆周长的方法 return 2*(this.area()/r);/使用this变量获取圆的面积,该程序的运行结果如下:圆的面积=78.53981633974999圆的周长,2super的使用场合super表示的是当前对象的直接父类对象,是当前对象的直接父类对象的引用。所谓直接父类,是相对于当前对象的其他“祖先”类而言的。例如,假设类A派生出子类B,类B又派生出自己的子类C,则B是C的直接父类,而A是C的祖先类。super代表的就是直接父类。若子类的数据成员或成员方法名与父类的数据成员或成员方法名相同,当要调用父类的同名方法或使用父类的同名数据成员时,可用关键字super来指明父类的数据成员和方法。,super的使用方法有三种:(1)用来访问直接父类隐藏的数据成员,其使用形式如下:super.数据成员(2)用来调用直接父类中被覆盖的成员方法,其使用形式如下:super.成员方法(参数)(3)用来调用直接父类的构造方法,其使用形式如下:super(参数)下面通过例子来说明前两种用法,关于第三种用法,将在节介绍。,【示例程序C5_13.java】super的使用。class A5 int x=4;int y=1;public void printme()System.out.println(“x=”+x+“y=”+y);System.out.println(“class name:”+this.getClass().getName();,public class C5_13 extends A5 int x;public void printme()int z=super.x+6;/引用父类(即A5类)的数据成员 super.printme();/调用父类(即A5类)的成员方法 System.out.println(“I am an”+this.getClass().getName();x=5;(“z=”+z+“x=”+x);/打印子类的数据成员,public static void main(String arg)int k;A5 p1=new A5();C5_13 p2=new C5_13();p1.printme();p2.printme();/super.printme();/错,在static方法中不能引用非static成员方法/k=super.x+23;/错,在static方法中不能引用非static数据成员,该程序的运行结果如下:x=4 y=1class name:A5x=4 y=1class name:c5_13I am an c5_13z=10 x=5,5.4.5 构造方法的重载与继承1构造方法的重载一个类的若干个构造方法之间可以相互调用。当一个构造方法需要调用另一个构造方法时,可以使用关键字this,同时这个调用语句应该是整个构造方法的第一条可执行语句。使用关键字this来调用同类的其他构造函数时,优点同样是可以最大限度地提高对已有代码的利用程度,提高程序的抽象度和封装性,减少程序的维护工作量。,【示例程序C5_14.java】构造方法的重载。class Addclass public int x=0,y=0,z=0;/以下是多个同名不同参数的构造方法 Addclass(int x)/可重载的构造方法1 this.x=x;Addclass(int x,int y)/可重载的构造方法2 this(x);/当前构造方法调用可重载的构造方法1 this.y=y;,Addclass(int x,int y,int z)/可重载的构造方法3 this(x,y);/当前构造方法调用可重载的构造方法2 this.z=z;public int add()return x+y+z;public class C5_14,public static void main(String args)Addclass p1=new Addclass(2,3,5);Addclass p2=new Addclass(10,20);Addclass p3=new Addclass(1);System.out.println(“x+y+z=”+p1.add();System.out.println(“x+y=”+p2.add();System.out.println(“x=”+p3.add();,该程序的运行结果如下:x+y+z=10 x+y=30 x=1,2构造方法的继承子类可以继承父类的构造方法,构造方法的继承遵循以下的原则:(1)子类无条件地继承父类的不含参数的构造方法。(2)如果子类自己没有构造方法,则它将继承父类的无参数构造方法,并将这些方法作为自己的构造方法;如果子类自己定义了构造方法,则在创建新对象时,它将先执行继承自父类的无参数构造方法,然后再执行自己的构造方法。(3)对于父类的含参数构造方法,子类可以通过在自己的构造方法中使用super关键字来调用它,但这个调用语句必须是子类构造方法的第一条可执行语句。,【示例程序C5_15.java】构造方法的继承。class Addclass2 public int x=0,y=0,z=0;Addclass2(int x)/父类可重载的构造方法1 this.x=x;Addclass2(int x,int y)/父类可重载的构造方法2 this.x=x;this.y=y;Addclass2(int x,int y,int z)/父类可重载的构造方法3 this.x=x;this.y=y;this.z=z;public int add()return x+y+z;,public class C5_15 extends Addclass2 int a=0,b=0,c=0;C5_15(int x)/子类可重载的构造方法1 super(x);a=x+7;C5_15(int x,int y)/子类可重载的构造方法2 super(x,y);a