JVM类加载(3)—初始化

3、初始化

在准备阶段,变量已经赋过一次系统要求的初始值,而在初始化阶段,则根据程序员通过程序制定的主观计划去初始化类变量(静态变量)和其他资源,或者从另外一个角度表达:初始化过程是执行类构造器<client>()方法的过程。<client>()方法:

  • <client>()方法是由编译器自动收集类中的所有类变量(静态变量)的赋值动作和静态语句块(static{})中的语句合并吃产生的,编译器收集的顺序是由语句在源文件中出现的顺序决定的,静态语句块只能访问到定义在静态语句块之前的变量,定义在它之后的变量,静态语句块能进行赋值操作,但是不能进行访问。
1 public class Test {
2     static{
3         i = 0;                 //这句能编译通过
4         System.out.println(i); //这里编译会报错,提示“非法向前引用”
5     }
6     static int i;
7 }
    • <client>()方法与类的构造函数不同,它不需要显式的调用父类构造器,虚拟机会保证在<client>()方法执行之前,父类的<client>()方法已经执行完毕。因此在虚拟机中第一个被执行的<client>()方法肯定是java.lang.Object的<client>()方法。
    • 由于父类的<client>()方法先执行,也就意味着父类中定义的静态语句块要优先于子类的变量赋值操作,如下代码,字段B的值会是2而不会是1
       1 public class Test {
       2     static class Parent{
       3         public static int A = 1;
       4         static{
       5             A = 2;
       6         }
       7     }
       8 
       9     static class Sub extends Parent{
      10         public static int B = A;
      11     }
      12     
      13     public static void main(String[] args) {
      14         System.out.println(Sub.B);
      15     }
      16 }

       

  • <client>()方法对于类或接口来说并不是必需的,如果一个类中没有静态语句块,也没有静态变量的赋值操作,那么编译器可以不为这个类生成<client>()方法。
  • 接口中不能使用静态语句块,但仍然有变量赋值初始化的操作,因此接口和类一样都会生成<client>()方法。但接口与类不同的是,执行接口的<client>()方法不需要执行父接口的<client>()方法,只有当父接口定义的变量被使用时父接口才会初始化。另外接口的实现进行初始化时,也不会执行接口的<client>()方法,同理除非访问了接口中定义的静态变量才会初始化接口。如果通过子类访问定义在父类中的静态变量时,只有父类会被初始化,子类则不会被初始化,如下代码清单:
     1 class Parent {
     2     public static int A = 1;
     3     static {
     4         System.out.println("this is parent");
     5     }
     6 }
     7 
     8 class Sub extends Parent {
     9     public static int B = 2;
    10     static {
    11         System.out.println("this is Sub");
    12     }
    13 }
    14 
    15 public class Test {
    16     public static void main(String[] args) {
    17         int i = Sub.A;
    18     }
    19 }
    20 // output:
    21 // this is parent
  • 虚拟机会保证一个类的<client>()方法在多线程环境中被正确的加锁、同步,如果多个线程去初始化一个类,那么只会有一个线程去执行这个类的<client>()方法,其他线程都需要阻塞等待,直到活动线程的<client>()方法执行完毕。如果一个类的<client>()方法有耗时很长的操作,就能造成多个进程阻塞,在实际应用中,这种阻塞往往是很隐蔽的。同一个虚拟机上类的<client>()方法只会执行一次。
  • 编译时的常量访问不会对该类进行初始化,如果只有在运行时才能确定,则会执行类的初始化动作,如下代码:
  •  1 import java.util.Random;
     2 
     3 class Init{
     4     static final int x = 6 / 3;
     5     static final int y = new Random().nextInt(100);
     6     static{
     7         System.out.println("init----");
     8     }
     9 }
    10 
    11 public class Test {
    12     public static void main(String[] args) {
    13         System.out.println(Init.x); //此处Init类不会进行初始化
    14         System.out.println(Init.y); //执行这条语句时,Init类才进行初始化
    15     }
    16 }
    17 // output:
    18 //2
    19 //init----
    20 //91

     

类只有在被首次主动使用时,才进行初始化,类的主动使用方式:

(1)、实例化一个类,new一个类的实例对象

(2)、访问类的静态变量

(3)、调用类的静态方法

(4)、通过反射调用类

(5)、实例化类的子类

(6)、被标位启动类的类

Java虚拟机执行类的初始化语句为类赋予初始值,在程序中静态变量初始化有两种方式

(1)、在变量声明处初始化

(2)、在静态代码块中进行初始化

1 private static int param1 = 1;//变量声明时初始化
2 private static int param2;
3 static{
4    param2 = 2; //静态代码块中进行初始化
5 }

 

针对private static int param1 = 1;在连接阶段的准备阶段时,param1变量被赋予int变量初始值0,在初始化阶段执行赋值操作,赋值为1。

posted @ 2017-05-07 20:54  七夜·雪  阅读(595)  评论(0编辑  收藏  举报