知乎网 问答: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
大功告成!!!
延伸阅读: