关于继承时子类重写父类方法和覆盖父类变量的若干问题 (待进一步研究)
2014-08-18 22:02
686 查看
假设,子类重载父类的方法,并将子类的成员覆盖。
创建子类对象实例,将其上转型成父类。
例子1
当执行1时结果为:
1 init parent
4 demo Son
3 init son
4 demo Son
当执行2时结果为:
1 init parent
4 demo Son
3 init son
4 demo Son
情况1和情况2其实是相似的,由于使用new Son()创建实例,上转型之后,访问的也是子类重载的方法而Parent中的init()方法中的this也被认为是指向堆区域中son的实例化对象,故this.demo()访问的仍然是子类中重载的方法。
例子2
当代码运行1时 我猜测结果应该是:
jack
jack
##jack
但实际结果却是:
tom
jack
## jack
此时Parent类中this.name指向的是父类中的name值
由于我们无法看到内存中的实际情况,但通过debug我们可以看到son对象中有两个同名的name变量
只有几种情况可以解释这种结果
1.this并不指向son在堆中创建的实例
2.创建对象时内存中使用了其他机制来保证这种结果的生成。
翻看了一下《深入理解Java虚拟机 JVM高级特性与最佳实践》这本书 似乎得出了一些端倪
由此可见当访问成员变量时,检测到父类的成员变量之后程序停止继续寻找。
对于方法,由于其内容存放在方法区内,每个对象的发放应该是通过其自身的this即引用唯一绑定。故就差不多解释的通了。
《深入理解Java虚拟机 JVM高级特性与最佳实践》一书的下载地址:
http://www.jb51.net/books/163531.html#down
上面只是个人的一点点猜测 ,不吝赐教。
创建子类对象实例,将其上转型成父类。
例子1
public class Parent { public void init() { System.out.println("1 init parent"); this.demo(); } public void demo() { System.out.println("2 demo parent"); } }
public class Son extends Parent { public void init(){ super.init(); System.out.println("3 init son"); this.demo(); } public void demo() { System.out.println("4 demo Son"); } public static void main(String[] args) { //Parent p = new Son();//1 Son son = new Son();//2 son.init(); // init(son) } }
当执行1时结果为:
1 init parent
4 demo Son
3 init son
4 demo Son
当执行2时结果为:
1 init parent
4 demo Son
3 init son
4 demo Son
情况1和情况2其实是相似的,由于使用new Son()创建实例,上转型之后,访问的也是子类重载的方法而Parent中的init()方法中的this也被认为是指向堆区域中son的实例化对象,故this.demo()访问的仍然是子类中重载的方法。
例子2
public class Parent { public String name="tom"; public void init() { System.out.println(this.name); } }
public class Son extends Parent { public String name="jack"; public void init(){ super.init(); System.out.println(this.name); } public static void main(String[] args) { //1 当前运行类 Son Son son = new Son(); son.init(); //init(son) System.out.println("## " + son.name); //2 当前运行类Parent Parent p = new Son(); p.init(); System.out.println("** " + p.name); } }
当代码运行1时 我猜测结果应该是:
jack
jack
##jack
但实际结果却是:
tom
jack
## jack
此时Parent类中this.name指向的是父类中的name值
由于我们无法看到内存中的实际情况,但通过debug我们可以看到son对象中有两个同名的name变量
只有几种情况可以解释这种结果
1.this并不指向son在堆中创建的实例
2.创建对象时内存中使用了其他机制来保证这种结果的生成。
翻看了一下《深入理解Java虚拟机 JVM高级特性与最佳实践》这本书 似乎得出了一些端倪
由此可见当访问成员变量时,检测到父类的成员变量之后程序停止继续寻找。
对于方法,由于其内容存放在方法区内,每个对象的发放应该是通过其自身的this即引用唯一绑定。故就差不多解释的通了。
《深入理解Java虚拟机 JVM高级特性与最佳实践》一书的下载地址:
http://www.jb51.net/books/163531.html#down
上面只是个人的一点点猜测 ,不吝赐教。
相关文章推荐
- 关于Java中子类能否继承和重写父类的静态方法问题
- java 子类继承父类成员变量的隐藏、实现方法的重写
- 关于子类继承父类后调用方法的问题
- Java编程关于子类重写父类方法问题的理解
- 关于父类的静态方法能否被子类重写的问题
- 关于子类继承父类后调用方法的问题
- 关于类继承中子类方法比父类方法丰富,一个引用值传递的小问题
- 解决重写父类的方法且不会影响继承的子类的问题
- OC中继承子类对象调用方法机制 子类对象访问父类中的实例变量
- Java 中子类是否只继承父类的非私有变量和方法?
- Java中的类成员访问控制符和继承重写父类的方法时的类成员访问控制符的相关问题
- Java编程中,子类可以用用这种方法给继承父类的成员变量赋值么?
- Java-关于子类是否继承了父类的private实例变量
- 子类重写父类的方法应注意的问题
- 深入理解子类继承父类中的成员变量问题
- java 子类重写父类的方法应注意的问题
- 关于在子类中重写父类的throws异常的方法的注意事项
- 关于子类调用父类构造方法的研究
- 子类继承父类(重写父类的静态方法,子类方法必须也是静态的)
- java中子类重写父类方法要注意的问题