这是一道坑人的面试题,看代码:
package com.sprone.classloader1;
public class ClassLoader1 {
public static void main(String[] args) {
Singleton singleton = Singleton.getInstance();
System.out.println("a= " + Singleton.a);
System.out.println("a= " + Singleton.b);
}
}
class Singleton {
private static Singleton singleton = new Singleton();// code1
public static int a;
public static int b = 0;
// code2
private Singleton() {
a++;
b++;
}
public static Singleton getInstance() {
return singleton;
}
}
打印结果是什么?把code1 的代码 剪切到code2 结果是什么?
答案:1,0 剪切后1,1
分析:类的加载 ,链接与初始化过程
加载:查找并加载类的二进制数据
链接:验证:确保被加载的类的正确性
准备:为类的静态变量分配内存,并将其初始化为默认值 比如private static int a = 3;在准备阶段分配内
存设置为0(int的默认值)
解析
问题1:对象对应的Class什么时候产生的? 假设是运行时,可想而知,JVM效率如蚂蚁,JVM团队也不会再运行时的时候才去生成Class对象,在我们编译之后.class文件里面都编译好了,每一个对象对应都有一个Class对象,而这个Class在字节码文件解析,为我们提供了一些生成对象的接口,我们常用的反射代码的生成都是在编译的时候生成的。
静态代码块顺序执行 a没有赋值,在准备阶段给其默认值。执行完code1 的时候 a为1; 继续下面的代码块,后面没有赋值,a=1,而b有重新赋值为0;
剪切后,new在后面 最终的值是在构造方法,所以都是1