反射机制必定用到Class.forName 这个方法。上周有个同事在分享这块的时候延展的谈到了Class.forName的一些用法。
1、Class.forName(String className);
2、Class.forName(String name,boolean initialize,ClassLoader loader);
通常大家都会认为,某个类中的static初始化区块都是在类加载 的时候就被调用到唯一的一次。真的是这样么?带着这个问题,我们来看看Class.forName这个方法。
Java代码
  1. public class Word {
  2. static{
  3. System.out.println(“Word Init….”);
  4. }
  5. }
  6. public class Test {
  7. public static void main(String[] args) {
  8. try{
  9. Class cls = Class.forName(“Word”,false,ClassLoader.getSystemClassLoader());
  10. }catch(Exception e){
  11. e.printStackTrace();
  12. }
  13. }
  14. }
  15. public class Test1 {
  16. public static void main(String[] args) {
  17. try{
  18. Class cls = Class.forName(“Word”,true,ClassLoader.getSystemClassLoader());
  19. }catch(Exception e){
  20. e.printStackTrace();
  21. }
  22. }
  23. }
public class Word {
	static{
		System.out.println("Word Init....");
	}
}
public class Test {
     public static void main(String[] args) {
		try{
			Class cls = Class.forName("Word",false,ClassLoader.getSystemClassLoader());
		}catch(Exception e){
			e.printStackTrace();
		}
	}
}
public class Test1 {
	public static void main(String[] args) {
		try{
			Class cls = Class.forName("Word",true,ClassLoader.getSystemClassLoader());
		}catch(Exception e){
			e.printStackTrace();
		}
	}
}
类Test和Test1运行的结果不一样:
Test:
Test1:Word Init…
为什么会这样?
Class.forName中的第二个参数在发生着作用。
不管用new还是class.forName来产生实例都隐含着两个部分:类加载+静态初始化区块调用;
静态初始化区块是在newInstance()的时候才被真正调用,所以Class.forName的第二个参数“false”的时候,定义不需要调用静态初始化区块,这就是为什么产生如上结果的原因了。
给一些应聘来工程师问这个问题,大家的回答多为静态初始化区块在类加载的时候被调用的。真正的是:静态初始区块并不是类加载的时候就调用,而是在第一次实例化的时候真正被调用