不灭的焱

革命尚未成功,同志仍须努力

作者:php-note.com  发布于:2019-02-27 23:29  分类:Java基础 

知乎网 问答:Java面向对象继承时出现的奇怪现象?

#父类#

package wen.jianbao.test;

public class Father {
   public String name = "我是父类的name";

    public void show() {
        System.out.println(this);
        System.out.println(this.name);
    }
}

#子类#

package wen.jianbao.test;

public class Demo {
    public static void main(String[] args) {
        Son mySon = new Son();
        mySon.show();
    }
}

#测试#

package wen.jianbao.test;

public class Demo {
    public static void main(String[] args) {
        Son mySon = new Son();
        mySon.show();
    }
}

输出:

wen.jianbao.test.Son@36baf30c
我是父类的name

如上示例,子类调用继承来的父类的方法,输出this,显示this是指向子类实例的指针。

但是输出this.name,却输出的是父类的成员变量,而没有使用子类中定义的同名变量(讲道理不应该就近选择子类的name变量?)

网友的回答:

只有方法才能被override,变量不行。调用Father.show()时,编译时已经把this.name绑定到Father.name了。

1. Son 类没有override Father 类 show()方法,因此,调用的是Father类的show()方法;
2. Father类的show()方法中第一个println()方法的参数为对象,属于运行时绑定,第二个println()方法的参数为Father类字段,属于编译时绑定

Son类并没有Override父类的show()方法,所以你在main中调用show()方法实际是调用的父类的方法,你可以在Son类中写一个show()方法覆盖父类的方法,再执行一下就是你期望的结果了。

 


思考:父类方法如何返回子类对象?

#父类#

package wen.jianbao.test;

public class Father {
    public String name = "我是父类的name";
    
    public Father setName(String name) {
        this.name = name;
        return this;
    }
}

#子类#

package wen.jianbao.test;

public class Son extends Father {
    String name = "我是子类的name";

    public void showName() {
        System.out.println(name);
    }
}

#测试#

package wen.jianbao.test;

public class Demo {
    public static void main(String[] args) {
        Son mySon = new Son();
        mySon.setName("wenjianbao").showName();
    }
}

报错:

Error:(6, 36) java: 找不到符号
  符号:   方法 showName()
  位置: 类 wen.jianbao.test.Father

很显然,子类调用的setName()方法,返回的是Father实例,它当然没有showName()方法了。。。但这不是我想要的效果!!!也就是回到了主题:父类方法如何返回子类对象实例?

解决方案:

使用泛型,传入父类,然后把this强制转换为父类,参考代码如下:

#父类#

package wen.jianbao.test;

// 关闭 在强制类型转换时编译器给出的 警告
@SuppressWarnings({"serial", "unchecked"}) 
public class Father<T extends Father> {
    public String name = "我是父类的name";
    
    public T setName(String name) {
        this.name = name;
        return (T)this;
    }
}

#子类#

package wen.jianbao.test;

public class Son extends Father<Son> {
    String name = "我是子类的name";

    public void showName() {
        System.out.println(name);
    }
}

#测试#

package wen.jianbao.test;

public class Demo {
    public static void main(String[] args) {
        Son mySon = new Son();
        mySon.setName("wenjianbao").showName();
    }
}

输出:

我是子类的name

大功告成!!!

 

 

延伸阅读:

关于◎SuppressWarnings("unchecked")