动手动脑(继承与多态)

1.为什么子类的构造方法在运行之前,必须调用父类的构造方法?能不能反过来?为什么不能反过来?

  构造方法主要用来创建对象时初始化对象,即为对象变量赋初始值。

  创建一个对象,先调用构造方法来进行初始化,子类拥有父类的成员变量和方法,如果不调父类的构造方法,则从父类继承而来的成员变量和方法得不到正确的初始化。不能反过来是因为父类不知道子类有什么变量,会导致程序出错。

2.现有三个类

class Mammal{}

class Dog extends Mammal{}

class Cat extends Mammal{}

先对其分别进行初始化:

Mammal m = null;

Dog d = new Dog();

Cat c = new Cat();

下列语句哪一个将引起编译错误?为什么?哪一个会引起运行时错误?为什么?

m = d;

d = m;

d = (Dog)m;

d = c;

c = (Cat)m;

出现错误:第二行,不能将父类变量赋给子类变量;

                  第四行,继承同一个父类的平行子类不能相互赋值;

                  第五行,在第一行中,m变量已经指向Dog对象,如果再执行c = (Cat)m,就是让c指向Dog对象,而Cat和Dog之间不存在继承关系,无法让c指向Dog对象。

class Mammal{}
class Dog extends Mammal {}
class Cat extends Mammal{}

public class TestCast
{
    public static void main(String args[])
    {
        Mammal m;
        Dog d=new Dog();
        Cat c=new Cat();
        m=d;
        d=m;
        d=(Dog)m;
        d=c;
        c=(Cat)m;

    }
}

测试截图:

3.运行以下测试代码

public class ParentChildTest {
    public static void main(String[] args) {
        Parent parent=new Parent();
        parent.printValue();
        Child child=new Child();
        child.printValue();
        
        parent=child;
        parent.printValue();
        
        parent.myValue++;
        parent.printValue();
        
        ((Child)parent).myValue++;
        parent.printValue();
        
    }
}

class Parent{
    public int myValue=100;
    public void printValue() {
        System.out.println("Parent.printValue(),myValue="+myValue);
    }
}
class Child extends Parent{
    public int myValue=200;
    public void printValue() {
        System.out.println("Child.printValue(),myValue="+myValue);
    }
}

(1)运行结果

(2)总结

    子类可以赋值给父类,代表父类变量引用子类的对象。子类可以覆盖父类,覆盖后引用的是子类的方法,如果需要引用父类的方法可以用super函数。当子类与父类拥有一样的方法,并且让一个父类变量引用一个子类对象时,到底调用哪个方法,由对象自己的“真实”类型所决定,这就是说:对象是子类型的,它就调用子类型的方法,是父类型的,它就调用父类型的方法。这个特性实际上就是面向对象“多态”特性的具体表现。如果子类与父类有相同的字段,子类方法中访问的是子类中的字段(而不是父类中的字段)。如果子类方法确实想访问父类中被隐藏的同名字段,可以用super关键字来访问它。如果子类被当做父类使用,则通过子类访问的字段是父类的。

posted @ 2018-11-02 08:55  苍天の笑  阅读(127)  评论(0编辑  收藏  举报