java static代码块执行时机

参考: https://www.cnblogs.com/jswang/p/7699643.html

一个类的运行分为以下步骤:

     1.装载

     2.连接

     3.初始化

    其中装载阶段又三个基本动作组成:

  1.     通过类型的完全限定名(绝对路径 : package.类名),产生一个代表该类型的二进制数据流
  2.     解析这个二进制数据流为方法区内的运行时数据结构
  3.     创建一个表示该类型的java.lang.Class类的实例

    另外如果一个类装载器在预先装载的时遇到缺失或错误的class文件,它需要等到程序首次主动使用该类时才报告错误。

 

    连接阶段(将java类的二进制代码合并到JVM的运行状态之中的过程)又分为三部分:

  1. 验证,确认类型符合Java语言的语义,检查各个类之间的二进制兼容性(比如final的类不用拥有子类等),另外还需要进行符号引用的验证。
  2. 准备,Java虚拟机为类变量分配内存,设置默认初始值。
  3. 解析(可选的) ,在类型的常量池中寻找类,接口,字段和方法的符号引用,把这些符号引用替换成直接引用的过程。

   

  当一个类被主动使用时,Java虚拟就会对其初始化(类构造器初始化),如下六种情况为主动使用:

  1. 当创建某个类的新实例时(如通过new或者反射,克隆,反序列化等)
  2. 当调用某个类的静态方法时
  3. 当使用某个类或接口的静态字段时
  4. 当调用Java API中的某些反射方法时,比如类Class中的方法,或者java.lang.reflect中的类的方法时
  5. 当初始化某个子类时
  6. 当虚拟机启动某个被标明为启动类的类(即包含main方法的那个类)

    Java编译器会收集所有的类变量初始化语句和类型的静态初始化器,将这些放到一个特殊的方法中:clinit() --> class init。

    虚拟机会保证一个类的 clinit()方法在多线程环境中被正确加锁和同步。

   附:

   类的被动引用(不会发生类的初始化)

  1.当访问一个静态变量时,只有真正声明这个变量的类百汇杯初始化(通过子类调用父类的静态变量不会导致子类初始化)

  2.通过数组定义类引用,不会触发此类的初始化(因为定义了一个类数组后,系统只是分配了一个引用空间,并没有实际分配内存空间给数组中的元素,因此类数组中的元素还是需要使用new运算符来实例化)

  3.引用常量不会触发此类的初始化(常量在编译阶段就存入调用类的常量池中了)


实际上,static块的执行发生在“初始化”的阶段。初始化阶段,jvm主要完成对静态变量的初始化,静态块执行等工作。

下面我们看看执行static块的几种情况:

1、第一次new A()的过程会打印"";因为这个过程包括了初始化

2、第一次Class.forName("A")的过程会打印"";因为这个过程相当于Class.forName("A",true,this.getClass().getClassLoader());

3、第一次Class.forName("A",false,this.getClass().getClassLoader())的过程则不会打印""。因为false指明了装载类的过程中,不进行初始化。不初始化则不会执行static块。

 

例子:

public class Test {

   static{
      System.out.println("静态初始化Test");
   }

   public static void main(String[] args) throws Exception {

      //被动引用
      System.out.println(A.MAX);
      A[] as = new A[10];
   }
}

class A extends A_Father {
   public static int width=100;   //静态变量,静态域    field
   public static final  int MAX=100;

   static {
      System.out.println("静态初始化类A");
      width=300;
   }
   public A(){
      System.out.println("创建A类的对象");
   }
}

class A_Father extends Object {
   static {
      System.out.println("静态初始化A_Father");
   }
}
结果为:静态初始化Demo01
       100

因为final常量是在连接阶段完成,并没有初始化。

你可能感兴趣的:(JVM)