课后作业4
运行 TestInherits.java 示例,观察输出,注意总结父类与子类之间构造方法的调用关系修改Parent构造方法的代码,显式调用GrandParent的另一个构造函数,注意这句调用代码是否是第一句,影响重大!
public class Grandparent { public Grandparent() { System.out.println("GrandParent Created."); } public Grandparent(String string) { System.out.println("GrandParent Created.String:" + string); } } public class Parent extends Grandparent { public Parent(){ //super("Hello.Grandparent.");该语句可用于调用Grandparent中的有参数的构造函数 System.out.println("Parent Created"); //super("Hello.Grandparent.");该语句必须写在子类构造方法的首句,否则编译会报错 } } public class Child extends Parent{ public Child(){ System.out.println("Child Created"); } } public class TestInherits { public static void main(String[] args){ Child c = new Child(); } }
运行结果(调用Grandparent默认构造函数):
运行结果(调用Grandparent中的有参数的构造函数):
注意:通过super调用基类构造方法,必须是子类构造方法的首句!
原因:构造方法的作用为初始化对象,必须先初始化父类,之后才能初始化子类对象。
参看ExplorationJDKSource.java
我们得到了一个奇特的运行结果:A@1c5f743版)
public class ExplorationJDKSource { /** * @param args */ public static void main(String[] args) { System.out.println(new A()); } } class A{}
A中没有任何成员,其继承来自于Object
main方法中实际上调用的是public void println(Object x)
这一方法内部调用了String类的valueOf方法
valueOf方法内部又调用了Object.toString方法:
public String toString() {
return getClass().getName()+"@"+Interger.toHexString(hashCode());
}
hashCode方法是本地方法,由JVM设计者实现
来看一段代码(示例Fruit.java ):
注意最后一句,一个字串和一个对象“相加”,得到以下结果:
在“+”运算中,当任何一个对象与一个String对象,连接时,会隐式地调用其toString()方法,默认情况下,此方法返回“类名 @ + hashCode”。为了返回有意义的信息,子类可以重写toString()方法。
在这个例子中正是Fruit类覆盖了Object类的toString方法,所以才会输出“Fruit toString”。