完整版见
https://jadyer.github.io/
package com.jadyer.classloader;
import java.util.Random;
/**
* 深入JVM之类的主动使用和final关键字
* @author 宏宇
* @editor Jan 25, 2012 2:01:47 PM
* @see =============================================================================
* @see 只有Java类被主动使用时,才会导致类被初始化。以下六种情况会导致类被主动使用
* @see 1、创建类的实例
* @see 2、访问某个类或接口的静态变量,或者对该静态变量赋值
* @see 3、调用类的静态方法
* @see 4、反射(如Class.forName("com.jadyer.classloader.FinalTest"))
* @see 5、初始化一个类的子类
* @see 6、Java虚拟机启动时被标明为启动类的类(java FinalTest或者右键Run as JavaApp)
* @see =============================================================================
*/
public class FinalTest {
static{
System.out.println("FinalTest static block");
}
public static void main(String[] args) {
System.out.println("==========================");
System.out.println(TestAA.xx);
System.out.println("==========================");
System.out.println(TestBB.xx);
System.out.println("==========================");
System.out.println(Child.aa); //这里实际只对父类进行了主动调用
Child.doSomething(); //在这个过程中,根本就没有对子类进行主动调用
}
}
/**
* 编译时就能计算出来"6/3"结果是2,也就是说xx是编译时的常量
* 所以FinalTest类在调用这里的xx时候,不会导致该类被初始化
* 注:这里所谓的不对类进行初始化和对类进行初始化,的根本在于
* 注:这里的静态代码块,不被执行和被执行
*/
class TestAA{
public static final int xx = 6 / 3;
static{
System.out.println("TestAA static block");
}
}
/**
* 只有在运行时才能确定随机数,故这里的xx不是编译时的常量
* 这时xx就是一个变量,即只有在运行的时候,才能确定它的值
* 所以FinalTest类在调用这里的xx时候,会导致该类初始化
*/
class TestBB{
public static final int xx = new Random().nextInt(100); //0-99之间的随机数
static{
System.out.println("TestBB static block");
}
}
class Parent{
static int aa = 6;
static{
System.out.println("Parent static block");
}
static void doSomething(){
System.out.println("Parent do Something");
}
}
class Child extends Parent{
static{
System.out.println("Child static block");
}
}