Java 多态方法构造器执行方法
我们参考下面这个例子:
读者可以提前考虑一下,这段程序的输出会是什么。
public class Polymorphism { /** * 创建一个类A * 该类中有一个方法draw,以及一个构造方法A */ static class A{ void draw(){ System.out.println("A.draw()"); } A(){ System.out.println("A() before draw()"); draw(); System.out.println("A() after draw()"); } } /** * 创建一个类B,继承A * 该类中同样有一个方法draw,以及一个构造方法B */ static class B extends A { private int value=1; void draw(){ System.out.println("B.draw(),value="+value); } B(int v){ value=v; System.out.println("B.B(),value="+value); } } /*现在我们调用B的构造函数,构造一个B*/ public static void main(String[] args) { new B(5); } }
最后的输出结果为
A() before draw() B.draw(),value=0 A() after draw() B.B(),value=5
初次分析
可见,当我们试图构造一个B时,应该会优先构造B的父类A,所以会调用父类A的构造函数A(),所以会输出
A() before draw()
这时A调用了draw()方法,因为是构造B类,而B类覆盖重写A类的draw()方法,所以这里应该调用的是B类重写过后的draw()方法,而B类的value默认值为1,所以会输出
B.draw(),value=1
A() after draw()
此刻父类A已经完成构造,所以接着才会构造B,调用B的构造函数B(),且传入的值为5,所以会输出
B.B(),value=5
而实际上,最后输出的并不是
B.draw(),value=1
而是
B.draw(),value=0
最后在《Thinking in Java》一书中找到了类似的例子,其中给出了一套正确的初始化顺序:
(1)在其他任何事物之前,将分配给对象的存储空间初始化为二进制的零;
(2)如前所述那样调用基类构造器。此时,调用被覆盖后的draw()方法 (要在调用B构造器之前调用),由于步骤1的缘故,我们此时会发现value的值为0。
(3)按照声明的顺序调用成员的初始化方法。
(4)调用导出类的构造器主体。
作者:Rekent
出处:http://www.cnblogs.com/rekent/
本文版权归作者和博客园共有,欢迎转载、点赞,但未经作者同意必须保留此段申明,且在文章页面明显位置给出原文链接,否则保留追究法律责任的权利。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 智能桌面机器人:用.NET IoT库控制舵机并多方法播放表情
· Linux glibc自带哈希表的用例及性能测试
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 手把手教你在本地部署DeepSeek R1,搭建web-ui ,建议收藏!
· 新年开篇:在本地部署DeepSeek大模型实现联网增强的AI应用
· Janus Pro:DeepSeek 开源革新,多模态 AI 的未来
· 互联网不景气了那就玩玩嵌入式吧,用纯.NET开发并制作一个智能桌面机器人(三):用.NET IoT库
· 【非技术】说说2024年我都干了些啥