为了搞清楚这个我们首先要知道一个类想要运行JVM会做哪些事情。
1、类装载
采用双亲委派模式加载类,子类会交给父类的classloader去加载,如果父类加载不到自己才会尝试加载。最终功能是将java字节码转换为JVM的class对象。
2、链接
将Java二进制代码合并到JVM的运行时状态中。在链接之前必须保证类已经被加载。期间会经过验证、准备和解析等几个步骤。验证确保java类的二进制表示在结构上是完全正确的,如果不正确抛出java.lang.VerifyError。准备过程则是创建类中的静态域并默认赋初值。解析的过程确保类引用的类能被找到。
3、初始化
当一个类真正被使用的时候,JVM会初始化该类。主要操作就是执行静态代码块和初始化静态域。
4、实例化
在内存中开辟堆空间。
从上我们可以看出,静态代码块在类初始化的时候执行。反应到代码上也就是在class.forname时执行。如下例子:
两个类Word:
package com.bjtest.belen;
public class Word{
static{
System.out.println("Word static initialization!");
}
public void start(){
System.out.println("Word starts");
}
}
Office:
package com.bjtest.belen;
public class Office{
public static void main(String args[])throws Exception{
args[0]="com.bjtest.belen.Word";
Office off = new Office();
System.out.println("类别准备载入");
Class c = Class.forName(args[0],true,off.getClass().getClassLoader());
System.out.println("类别准备实例化");
Class c1 = Class.forName(args[0],true,off.getClass().getClassLoader());
Object o = c.newInstance();
Object o2= c.newInstance();
}
}
执行代码结果如下:
类别准备载入
Word static initialization!
类别准备实例化
从上分析得出如下结论,类只被加载一次。并且静态代码块只在类初始化的时候被执行一次,第二次不会执行。
但是如果将上述代码改为:
package com.bjtest.belen;
public class Office{
public static void main(String args[])throws Exception{
args[0]="com.bjtest.belen.Word";
Office off = new Office();
System.out.println("类别准备载入");
Class c = Class.forName(args[0],false,off.getClass().getClassLoader());
System.out.println("类别准备实例化");
Class c1 = Class.forName(args[0],true,off.getClass().getClassLoader());
Object o = c.newInstance();
Object o2= c.newInstance();
}
}
执行结果如下:
类别准备载入
类别准备实例化
Word static initialization!
从上结合JDK分析我们可以得出如下结论:
1、我们可以控制类加载链接时不初始化
2、类在实例化时如果没有初始化,那么触发初始化。
3、静态代码块只执行一次,并且只在初始化时执行