类初始化(静态方法及静态域)

注意,有一点很有意思,当使用“.class”来创建对Class对象的引用时,不会自动地初始化该Class对象。

 

为了使用类而做的准备工作实际包含三个步骤:

1.加载 。这是由类加载器执行的。该步骤将查找字节码,并从这些字节码中创建一个Class对象。

2.链接 。这个阶段将验证类中的字节码,为静态域分配存储空间,并且如果必需的话,将解析这个类创建的对其他类的所有引用。

3.初始化 。如果该类具有超类,则对其进行初始化,执行静态初始化器和静态初始化块。

 

 

初始化被延迟到对静态方法(构造器隐式地是静态的)或者非常数静态域进行首次引用时才执行:

 

import java.util.*;

class Initable {
  static final int staticFinal = 47;
  static final int staticFinal2 =
    ClassInitialization.rand.nextInt(1000);
  static {
    System.out.println("Initializing Initable");
  }
}

class Initable2 {
  static int staticNonFinal = 147;
  static {
    System.out.println("Initializing Initable2");
  }
}

class Initable3 {
  static int staticNonFinal = 74;
  static {
    System.out.println("Initializing Initable3");
  }
}

public class ClassInitialization {
  public static Random rand = new Random(47);
  public static void main(String[] args) throws Exception {
    Class initable = Initable.class;
    System.out.println("After creating Initable ref");
    // Does not trigger initialization:
    System.out.println(Initable.staticFinal);
    // Does trigger initialization:
    System.out.println(Initable.staticFinal2);
    // Does trigger initialization:
    System.out.println(Initable2.staticNonFinal);
    Class initable3 = Class.forName("Initable3");
    System.out.println("After creating Initable3 ref");
    System.out.println(Initable3.staticNonFinal);
  }
} 

/* Output:
After creating Initable ref
47
Initializing Initable
258
Initializing Initable2
147
Initializing Initable3
After creating Initable3 ref
74
*///:~

 

1.从initable引用的创建中可以看到,仅使用.class语法来获得对类的引用不会引发初始化。

 

2.Class.forName()的方法立即进行了初始化,如initable3的引用的创建。

 

3.如果一个static final值是“编译期常量”,就像Initable.staticFinal,那么这个值不需要对Initable类进行初始化就可以被读取。

 

4.如果只是将一个域设置为static final,还不足以确保这种行为,例如,对Initable.staticFinal2的访问将强制进行类的初始化,因为它不是一个编译期常量。

 

5.如果一个static域不是final的,那么对它访问时,总是要求在它被读取之前,要先进行链接(为这个域分配存储空间)和初始化(初始化该存储空间   ),就像对Initable2.staticNonFinal的访问所看到的那样。

你可能感兴趣的:(Java,Details)