课后作业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”。