先看两个单例:
编写main方法调用单例:
得到结果:
之所以结果不一样,是因为java创建实例的原因,java创建实例的过程是:
1.首先去JVM 的方法区中区寻找类的class对象,如果能找到,则按照定义生成对象,找不到则转22.加载类定义:类加载器(classLoader)寻找该类的 .class文件,找到后对文件进行分析转换为class对象存入方法区方便以后调用。 其中jdk 的class一般是在jvm启动时用启动类加载器完成加载,用户的class则是在用到的时候再加载。3.在jvm的堆中给对象开辟一个内存空间4.对象初始化,顺序:(1) 父类静态对象(如果有对对象赋值,则付初始值),静态代码块 例如 private static Singleton singleton = new Singleton();就是先加载Singleton 类,然后对singleton 赋值,调用new Singleton()(2)子类静态对象,静态代码块 ----(1)和(2)是类加载的时候执行。(3)父类非静态对象,非静态代码块(按代码先后顺序)(4)父类构造函数(5)子类非静态对象,非静态代码块(6)子类构造函数
在第一个单例的创建过程是:
public class Singleton { private static Singleton singleton = new Singleton(); public static int i; public static int y = 0;
private Singleton() { i++; y++;}
当这个Singleton第一次被加载的时候,会先执行private static Singleton singleton = new Singleton();,会去加载左边Singleton这个类,但是这个类与当前的Singleton是同一个类,已经在加载的过程。然后执行等号右边对singleton属性赋值,就是执行new Singleton(),进入Singleton的无参构造器private Singleton() { i++; y++;},当执行到i++时,发现属性i还没有被加载,于是就先加载属性i,因为是int类型,所以默认值是0,执行i++后,i的值为1
同理,y一样被加载,此时y值也为1。当执行完构造函数,就将执行结果赋给属性Singleton
当 private static Singleton singleton = new Singleton();执行完后,程序走到public static int i,发现i已经在构造函数里面被加载过,就执行下一行static int y=0;
此时的y已经是被加载过,但是还没将 = 0执行,于是就将0赋给y。
所以第一个的结果为1,0
而第二个单例中,原理也一样
class Singleton2 { public static int i; public static int y = 0; private static Singleton2 singleton = new Singleton2(); private Singleton2() { i++; y++; }
首次加载Singleton2会先加载i,在代码里面没有给i赋值,但是int 类型初试值为0,所以i的值为0
然后加载y并给y赋值 y=0。
然后加载Singleton2这个类(因为是同一个类,已经在加载了,所以不会再去加载。如果这个是不同的类,则会先去加载这个类),然后执行new Singleton2(),调用Singleton2的构造函数,给i和y的值加1。然后把实例赋给singleton2这个属性。
所以singleton2 的执行结果是
i=1 ,y=1