深入java虚拟机之类的静态代码块执行时机

public class Test {
  public static void main(String[] args) throws ClassNotFoundException {
    // System.out.println(Class.forName("java.lang.String").getClassLoader());
    // System.out.println(Class.forName("java.lang.Object").getClassLoader());
    // String str1 = new String("123");
    // String str2 = new String("456");
    // System.out.println(str1.getClass() == str2.getClass());
    System.out.println(Test1.x);
    System.out.println("------------");
    System.out.println(Test2.x);
  }
}
class Test1 {
  public static final int x = 6 / 3;
  static {
    System.out.println("Test1");
  }
}
class Test2 {
  public static final int x = new Random().nextInt(100);
  static {
    System.out.println("Test2");
  }
}

执行结果如下:

2
------------
Test2
50

为什么?

静态代码块在类初始化的时候才会执行,所以首先明白类的初始化时机:

– 创建类的实例
– 访问某个类或接口的静态变量,或者对该静态变量赋值
– 调用类的静态方法
– 反射(如Class.forName(“com.shengsiyuan.Test”))
– 初始化一个类的子类
– Java虚拟机启动时被标明为启动类的类( JavaTest)

注意第二点,– 访问某个类或接口的静态变量,或者对该静态变量赋值

Test1和Test2加了final关键字,都是常量为什么还初始化类了呢

Test1中6/3在编译的时候就知道值了,所以x是静态常量,所以Test1是不会初始化的。而Test2的x是运行的时候才能知道结果。所以Test2会初始化。

如果都去掉final关键字,那么都会2个类都会进行类的初始化,都会打印出东西。就符合第二点了。

 

你可能感兴趣的:(深入java虚拟机之类的静态代码块执行时机)