我们先来看一个例子,如果你读过《java编程思想》的话 应该会有印象

  1. package com.test.zj;
  2.  
  3. public class PolyConstructors {
  4.  
  5. public static void main(String[] args) {
  6. // TODO Auto-generated method stub
  7. new RoundGlyph(5);
  8. }
  9.  
  10. }
  11.  
  12. class RoundGlyph extends Glyph {
  13. private int radius = 1;
  14.  
  15. public RoundGlyph(int r) {
  16. // TODO Auto-generated constructor stub
  17. radius = r;
  18. System.out.println("RoundGlyph radius==" + radius);
  19. }
  20.  
  21. @Override
  22. void draw() {
  23. // TODO Auto-generated method stub
  24. System.out.println("RoundGlyph draw() radius==" + radius);
  25. }
  26.  
  27. }
  28.  
  29. class Glyph {
  30. void draw() {
  31. System.out.println("print glyph.draw()");
  32. }
  33.  
  34. Glyph() {
  35. System.out.println("Glyph() before draw()");
  36. draw();
  37. System.out.println("Glyph() after draw()");
  38.  
  39. }
  40.  
  41. }

对于java基础一般的同学来说 这里你可能会认为输出是如下:

  1. Glyph() before draw()
  2. RoundGlyph draw() radius==1
  3. Glyph() after draw()
  4. RoundGlyph radius==5

但实际上你运行完毕以后 你会发现他的输出是这样的:

可能有的人读到这里还是不太明白我要表述什么,那我再写一个简单的例子。先定义一个父类SuperClass

  1. package com.test.zj;
  2.  
  3. public class SuperClass
  4. {
  5. private int superValue;
  6.  
  7. public SuperClass()
  8. {
  9. setSuperValue(100);
  10.  
  11. }
  12.  
  13. public void setSuperValue(int x)
  14. {
  15. superValue=x;
  16. }
  17.  
  18. }

然后我们定义它的子类:

  1. //这个子类继承自父类superclass
  2. public class SubClass extends SuperClass
  3. {
  4. private int subValue=10;
  5.  
  6. public SubClass()
  7. {
  8.  
  9. }
  10. //这个方法重写了父类的方法
  11. public void setSuperValue(int x)
  12. {
  13. //先调用父类的方法
  14. super.setSuperValue(x);
  15. //然后把值赋给自己的变量
  16. subValue=x;
  17.  
  18. }
  19.  
  20. public void printSubValue()
  21. {
  22. System.out.println("subclass subvalue=="+subValue);
  23. }
  24.  
  25. }

最后写个main函数 就可以了

  1. package com.test.zj;
  2.  
  3. public class MainClass {
  4.  
  5. public static void main(String[] args) {
  6. // TODO Auto-generated method stub
  7. SubClass sc=new SubClass();
  8. sc.printSubValue();
  9. }
  10.  
  11. }

好,现在我相信很多人都会认为第二个例子输出的结果应该是100

但其实并没有什么卵用,他的实际结果是:

那到底这两个例子都发生了什么呢,我们直接来看字节码好了,这个字节码肯定不会有错,字节码怎么写的 jvm就怎么执行。

我们就先看看第一个例子。

这里应该很明显的能看到 我们的main函数 一开始就是new了RoundGlyph这个对象。那我们看看这个类-c的结果吧

可以看到这个类的构造函数

先执行的是这个:

也就是说 先执行了glyph的构造方法 然后当glyph的构造函数执行完毕以后 才执行的赋值语句

我们的radius 作为一个int变量 在被执行之前 jvm自动初始化他的值为0!

所以你这里隐隐约约应该都能猜到一个大概了,先执行的glyph的 构造函数,然后再给自己的成员变量radius赋值。

那我们看看glyph 都做了什么吧:

你看glyph的构造函数, 在中间的时候13:invokevirtual #31 这里,去执行了draw方法,但是子类我们重写了这个draw方法

所以你看 在glyph的构造函数里 调用子类的draw方法的时候 子类的radius赋值语句并没有被执行到,所以子类的这个方法

输出的值当然是0!

当父类glyph的构造函数执行完毕以后 ,我们的子类的赋值语句才终于得到执行。所以到这里 你应该能明白第一个例子了。

那我们现在就可以去研究一下第二个例子,其实都是大同小异的。我们还是先看第二个例子的manclass和main函数

你看这里main函数 先是new了一个subclass 子类的对象 对吧。那我们当然就要去看看subclass init方法

实际上这个地方就是Subclass的构造函数了。

这里很清楚的可以看到 在subclass的构造函数里 我们是先执行的superclass的构造函数,然后才给自己的subValue赋值为10.

那我们就去看看superclass里都做了什么。但实际上走到这里我们已经能想到了无论你在superclass做了什么 当你做完以后

subValue的值都必定为10.

所以当你subclass的对象构造完毕以后 此时他的成员变量subvalue的值就是10了,所以你当然打印出来这个变量的值 就一定是10了。

当然为了更清晰一点 我还是把superclass构造函数里做了什么稍微讲一下,虽然这里面做了什么不会影响到我们的结论,但还是讲一下吧,

即使这并没有什么卵用。。。

你看这里就是调用了一下setSuperValue这个方法么,对吧,因为子类重写了这个方法 所以我们肯定要看看子类

这个方法干嘛的:

你看不就是又调用了父类的setSupervalue方法吗,然后调用以后 你看有个iload putfield

这2个操作不就是给我们子类的subvalue 赋值的吗,对吧。一直到这里,我们子类的对象构造函数的第一步:

调用父类的构造函数 就算是走完了,走完了以后 才终于执行了自己的赋值语句:

好,这2个例子到这里就算分析完毕了。

实际上最终的结论就是java编程思想里说的那样:

父类static成员 -> 子类static成员 -> 父类普通成员初始化和初始化块 -> 父类构造方法 -> 子类普通成员初始化和初始化块 -> 子类构造方法

如果你们有兴趣的话,可以写一个稍微更复杂一点的程序,验证一下 上面的这个结论是否成立,废话。。。。这结论肯定是成立的。但是

你如果用javap -c 这个命令 去看他们的字节码的话 相信你能理解的更深了!

最后多说一句,平常我们在写代码的时候,尽量避免 上述2个例子这样的写法,因为这种情况造成的bug 很难被发现。。。即:

尽量不要在父类的构造函数里  操作子类的成员变量。如果一定要把初始化写的很麻烦的话,请考虑使用初始化块 这样一目了然的方法!

别问我为什么会研究到这,因为tmd 有一个bug 找了好久 发现是这个原因啊!所以以后你们发现有人这么写,请直接写邮件抄送全组投诉他啊!

java 构造函数内部的多态方法 完全剖析的更多相关文章

  1. [Java]构造函数内部多态的行为所引起的灾难

    构造函数内部的多态行为所产生的意想不到的结果 一.Java版本 1 package com.company; 2 import static com.brianyi.util.Print.*; 3 4 ...

  2. Java编程思想之八多态

    在面向对象的程序设计语言中,多态是继数据和继承之后的第三张基本特征 多态不但能够改善代码组织结构和可读性,还能够创建可扩展的程序--即无论在项目最初创建时还是在需要添加新功能时都可以"生长& ...

  3. 《java编程思想》多态与接口

    向上转型 定义:把某个对象的引用视为对其基类类型的引用的做法被称为向上转型方法调用绑定 将一个方法调用同一个方法主体关联起来被称作绑定. 前期绑定:程序执行前进行的绑定叫做前期绑定,前期绑定也是jav ...

  4. java基础篇 之 构造器内部的多态行为

    java基础篇 之 构造器内部的多态行为 ​ 我们来看下下面这段代码: public class Main { public static void main(String[] args) { new ...

  5. JAVA构造函数(方法)

    一.什么是构造函数 java构造函数,也叫构造方法,是java中一种特殊的函数.函数名与相同,无返回值. 作用:一般用来初始化成员属性和成员方法的,即new对象产生后,就调用了对象了属性和方法. 在现 ...

  6. Java中的多态方法

    public class Main { public void test(Object o) { System.out.println("Object"); } public vo ...

  7. java 实验3 继承+多态

    实验3  继承与多态 **类可以实现多个接口 但只有单继承!** 1.继承 1).继承语法  class 子类名 extends 父类{    } 2).构造函数(通过source即可得到) 注意: ...

  8. java提高篇之理解java的三大特性——多态

    面向对象编程有三大特性:封装.继承.多态. 封装隐藏了类的内部实现机制,可以在不影响使用的情况下改变类的内部结构,同时也保护了数据.对外界而已它的内部细节是隐藏的,暴露给外界的只是它的访问方法. 继承 ...

  9. [C#解惑] #1 在构造函数内调用虚方法

    谜题 在C#中,用virtual关键字修饰的方法(属性.事件)称为虚方法(属性.事件),表示该方法可以由派生类重写(override).虚方法是.NET中的重要概念,可以说在某种程度上,虚方法使得多态 ...

随机推荐

  1. 机器学习之单变量线性回归(Linear Regression with One Variable)

    1. 模型表达(Model Representation) 我们的第一个学习算法是线性回归算法,让我们通过一个例子来开始.这个例子用来预测住房价格,我们使用一个数据集,该数据集包含俄勒冈州波特兰市的住 ...

  2. Struts2.0 去掉action后缀名

    刚刚接触Struts2.0,发现默认请求都会带着后缀名:action 就如下图,url地址中会暴露login.action(请原谅struts拼写错误..) 作为一个URL简洁爱(chu)好(nv)者 ...

  3. java多线程浅谈

    当一个线程进入一个对象的一个synchronized方法后,其它线程是否可进入此对象的其它方法? 分这几种情况:    1.其他方法前是否加了synchronized关键字,如果没加,则能.    2 ...

  4. <iostream> 和 <iostream.h>的区别 及 Linux下编译iostream.h的方法

    0.序言 其实2者主要的区别就是iostream是C++标准的输入输出流头文件,而iostream.h是非标准的头文件. 标准头文件iostream中的函数属于标准命令空间,而iostream.h中的 ...

  5. 一个简单的以User权限启动外部应用程序(用NetUserAdd函数和USER_INFO_1结构体动态添加用户,然后用CreateProcessWithLogonW启动程序)

    版权声明:本文为博主原创文章,未经博主允许不得转载. BOOL ExecuteAsUser(LPCWSTR lpszUserName, LPCWSTR lpszPassword, LPCWSTR lp ...

  6. MakeObjectInstance的前世今生(关键是ECX的何时入栈以及Self指针何时存储的)

    高手们的文章有很大启发,但是总有些小错,也有没交代清楚的,以下是我的理解: 编译器编译MainWndProc的时候,它是一个正常Delphi普通函数,MakeObjectInstance对它做变换是运 ...

  7. 复习一下,? extends T 和 ? super T

    前话 最近学一些杂七杂八的东西,都把基础给忘了. 比如Java泛型中的 ? extends T和 ? super T 吧. 刚看开源项目的时候遇到它,表情如下: 源码分析 直接用源码来讲解吧 pack ...

  8. Android笔记——RecyclerView替代ListView

    ListView是常用列表控件,但设置Adapter时自定义代码较为复杂,因此Android3.0后,增加RecyclerView替代ListView RecyclerView没有提供OnItemCl ...

  9. 正确的理解this 和 super

    this和super是Java的两个关键字. 先明确一个问题,有人错误的认为它们是对象里的“属性”,这只能怪老师没有讲清楚计算机的本质了.因为计算机的处理器只能用指令去处理数据,像C语言之类的容易理解 ...

  10. Java:IO流之转换流

    IO流的基本流情况: 字符流:                字符流缓冲区: FileReader              BufferedReader FileWriter             ...