我最近遇到了这些问题,但在StackOverflow上找不到答案。

  • Java类变量以什么顺序初始化?
  • 还有一些相关的问题,是否可以对变量重新排序以更改类行为?
  • 为什么?

  • 如对Meta的建议,我将发布对这个问题的答案。

    最佳答案

    在Java中,类变量在following order中初始化:

  • 父类(super class)的静态变量
  • 此类的所有静态变量均设置为其default values
  • 静态变量和静态初始化块,按声明顺序。
  • 父类(super class)的实例变量
  • 此类的所有实例变量都设置为其default values
  • 实例变量和实例级别初始化块,按声明顺序

  • 1和2仅在实例化类的第一次时执行。

    因此,给出以下代码:
    class Test
      extends TestSuper
    {
      final int ti1;
      final int ti2 = counter ++;
      { ti1 = counter ++; }
      static final int ts1;
      static final int ts2 = counter ++;
      static { ts1 = counter ++; }
    
      public static void main(String[] argv) {
        Test test1 = new Test();
        printTest(test1);
        Test test2 = new Test();
        printTest(test2);
      }
      private static void printTest(Test test) {
        System.out.print("ss2 = " + test.ss2);
        System.out.print(", ss1 = " + test.ss1);
        System.out.print(", ts2 = " + test.ts2);
        System.out.println(", ts1 = " + test.ts1);
        System.out.print("si2 = " + test.si2);
        System.out.print(", si1 = " + test.si1);
        System.out.print(", ti2 = " + test.ti2);
        System.out.println(", ti1 = " + test.ti1);
        System.out.println("counter = " + test.counter);
      }
    }
    
    class TestSuper
    {
      static int counter = 0;
      final int si1;
      final int si2 = counter ++;
      { si1 = counter ++; }
      static final int ss1;
      static final int ss2 = counter ++;
      static { ss1 = counter ++; }
    }
    

    然后我们得到以下输出:
    ss2 = 0, ss1 = 1, ts2 = 2, ts1 = 3
    si2 = 4, si1 = 5, ti2 = 6, ti1 = 7
    counter = 8
    ss2 = 0, ss1 = 1, ts2 = 2, ts1 = 3
    si2 = 8, si1 = 9, ti2 = 10, ti1 = 11
    counter = 12
    

    从此输出中,我们可以看到字段是按照列表中指定的顺序初始化的。

    现在,关于第二个问题,可以对字段重新排序以更改类行为。是的,通过对字段重新排序,您可以更改字段的初始化顺序。现在,在所有字段都是独立的特定情况下,这不会影响观察到的行为,但是,只要字段不是独立的,例如在上面的代码中,那么重新排序字段就可以更改其初始化值。

    例如,如果三行:
      static final int ss1;
      static final int ss2 = counter ++;
      static { ss1 = counter ++; }
    

    更改为:
      static final int ss1;
      static { ss1 = counter ++; }
      static final int ss2 = counter ++;
    

    然后输出将更改为:
    ss2 = 1, ss1 = 0, ts2 = 2, ts1 = 3
    si2 = 4, si1 = 5, ti2 = 6, ti1 = 7
    counter = 8
    

    也就是说,ss2ss1将更改值。

    原因是在Java Language Specification中指定了此行为。

    关于java - Java类变量以什么顺序初始化?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/2138801/

    10-11 02:42
    查看更多