Android学习笔记——Java各种单例模式及区别
1、经典懒汉
public class Danli {
private volatile static Danli danli1;
public static Danli getInstance() {
if (danli1 == null) {
synchronized (Danli.class) {
if (danli1 == null) {
danli1 = new Danli();//111
}
}
}
return danli1;
}
}
synchronized加锁保证了只有一个线程能够执行到代码块里面
代码块里面再次进行非空判断,主要是因为:假如线程A、B同时执行到第一个if判断里面,表明在A、B两个线程里面danli1都为null,此时A、B两个线程进行抢锁,A线程拿到了锁,B线程被阻塞在外面,如果没有第二个if判断,那么当A线程执行完代码块之后new Danli();,然后释放锁,B线程拿到锁之后实际上B线程还是处于danli1==null的状态,那么此时有if判断,则B线程会直接跳出。
需要volatile关键字的原因是,在并发情况下,如果没有volatile关键字,在注释111会出现问题。
danli1 = new Danli();可以分解为3行伪代码
a、memory = allocate() //分配内存
b、ctorInstanc(memory) //初始化对象
c、danli1 = memory //设置instance指向刚分配的地址
上面的代码在编译运行时,可能会出现重排序从a-b-c排序为a-c-b。在多线程的情况下会出现以下问题。当线程A在执行第5行代码时,B线程进来执行到第2行代码。假设此时A执行的过程中发生了指令重排序,即先执行了a和c,没有执行b。那么由于A线程执行了c导致danli1指向了一段地址,所以B线程判断danli1不为null,会直接跳到第6行并返回一个未初始化的对象。
2、饿汉
public class Danli {
private Danli() {
}
private static Danli danli2 = new Danli();
}
不管用与不用,反正只要这个类被加载了,就直接new出实例。
3、懒汉
public class Danli {
private static class GetIns {
private static Danli danli3 = new Danli();
}
private static Danli getDanli3() {
return GetIns.danli3;
}
}
通过静态内部类进行实例化
饿汉式:简单来说就是空间换时间,因为上来就实例化一个对象,占用了内存,(也不管你用还是不用)
懒汉式:简单的来说就是时间换空间,与饿汉式正好相反