成员变量赋值问题【向前引用】

今天,同窗问了我一个问题,代码以下html

public class Test10 {

    //这个小程序 为啥不报错,应该是先执行a=10这个程序块,可是咱们没有设定a的类型啊
    {
        a = 10;
    }

    int a = 20;


    public static void main(String[] args) {
        Test10 t = new Test10();
        System.out.println("a的数值为:" + t.a);
    }
}
  • 开始的时候,我看到这个代码有点懵了(卧槽,谁会这样用啊),因而我就在想:为何不报错,是否是JVM会把代码全都编译一次,执行到构造代码块的时候,发现后面有对a变量进行定义,因此没有报错。因而就上网查资料去了,后来发现java支持向前引用
  • 接着,我发现输出的值是20(卧槽,怎么是20???,不该该是a=10构造代码块覆盖了在定义成员变量时a=20吗)
  • 因而乎,我又做死地加入了一个构造函数进去,看看执行顺序究竟是怎么样的
public class Test10 {


    public Test10() {

        a=30;
    }

    //这个小程序 为啥不报错,应该是先执行a=10这个程序块,可是咱们没有设定a的类型啊
    {
        a = 10;
    }

    int a = 20;


    public static void main(String[] args) {
        Test10 t = new Test10();
        System.out.println("a的数值为:" + t.a);
    }
}
  • 由于java支持向前引用,构造函数的a那固然不会报错啦。我再次执行该测试的时候,发现a的值为30,此时又懵逼了。按我当时理解是这样的(既然上面输出的是20,那么我加入构造方法,构造方法在构造代码块后执行,一样会被a=20覆盖掉【当时候脑壳短路了,忘记构造方法会覆盖掉成员变量的值】
  • 最后我在知乎RednaxelaFX—-R大中一个回答中彷佛找到了答案。截图以下:

  • 简要提炼下语句:对JVM来讲全部实例初始化动做都要收集到“特殊的实例初始化方法”(名为“init”,内容对应全部实例初始化器+构造器)里,按代码顺序把实例初始化动做(包括实例字段初始化与匿名的实例初始化器)收集起来,而后是构造器自身的内容
  • 就是说构造代码块和成员变量的赋值顺序是依照代码的顺序执行的
  • 剔除构造方法,将构造代码块和声明成员变量位置交换一下,看看结果
public class Test10 {


    int a = 20;

    {
        a = 10;
    }


    public static void main(String[] args) {
        Test10 t = new Test10();
        System.out.println("a的数值为:" + t.a);
    }
}
  • 输出的结果是10。java

  • ps:若是我有理解错误的地方,请留言,谢谢!小程序

参考资料:markdown

向前引用:http://www.cnblogs.com/nokiaguy/p/3156357.html函数

执行顺序RednaxelaFX的回答:https://www.zhihu.com/question/36643366?q=java%E6%88%90%E5%91%98%E5%8F%98%E9%87%8F%E5%88%B0%E5%BA%95%E4%BB%80%E4%B9%88%E6%97%B6%E5%80%99%E8%A2%AB%E8%B5%8B%E5%80%BC%EF%BC%9F测试

相关文章
相关标签/搜索