单例模式概括

单例

单例模式是一种对象的创建模式,用于产生一个对象的具体实例,他可以确保系统中,只有一个实例,减少new的次数,因而对系统内存的使用频率也会降低,减轻GC的压力,缩短GC停顿的时间

恶汉式单例

public class HungrySingle {
    private final static HungrySingle SINGLE = new HungrySingle();

    private HungrySingle() {
        System.out.println("HungrySingle is init");
    }

    public static HungrySingle instance() {
        return SINGLE;
    }

}
  • 此种方式,是在虚拟机加载该类的时候,就加载该类的实例,无法对实例进行延时的加载

懒汉模式

public class LazySingle {
    public static LazySingle SINGLE = null;

    private LazySingle() {
    }

    public static LazySingle instance() {
        if (null == SINGLE) {
            SINGLE = new LazySingle();
        }
        return SINGLE;
    }
}
  • 优点:保证了延时的加载
  • 缺点:不能保证线程的安全,容易重复创建多个实例,在多线程的情况下,这种模式是失效的,无法保证单个实例

懒汉线程安全

public class LazySingle {
    public static LazySingle SINGLE = null;

    private LazySingle() {
    }

    //一种方式
    public static synchronized LazySingle instance01(){
        if (null==SINGLE){
            SINGLE = new LazySingle();
        }
        return SINGLE;
    }

    //二种方式
    public static LazySingle instance02() {
        synchronized (LazySingle.class) {
            if (null == SINGLE) {
                SINGLE = new LazySingle();
            }
            return SINGLE;
        }
    }
}

该方式保证了线程安全,但有点影响性能,

懒汉Dcl模式

public class LazySingle {
    public static LazySingle SINGLE = null;

    private LazySingle() {
    }
    public static LazySingle instance03() {
        //避免了不必要的同步
        if (null == SINGLE) {
            synchronized (LazySingle.class) {
                //初次的时候初始化对象
                if (null == SINGLE) {
                    SINGLE = new LazySingle();
                }
            }
        }
        return SINGLE;
    }
}
  • 优点: 解决了性能问题,和线程安全的问题
  • 缺点:1、在理想状态下,jvm先给对象分配内存,调用构造方法初始化对象,将对象指向分配的内存空间。jvm在及时编译器中有指令重排序的优化,不会完全按照这个步骤去创建对象,这个就造成了线程不安全的问题。

懒汉DCL优化

  • 解决办法就是添加关键字 volatile 保证线程在本地不会存在instance的副本,每次都会到内存中去读取. 使用该关键字,就是禁用了jvm指令重排序优化,避免造成线程安全的问题
public class LazySingle {
    //添加 volatile关键字
    public static volatile LazySingle SINGLE = null;

    private LazySingle() {
    }

    public static LazySingle instance03() {
        //避免了不必要的同步
        if (null == SINGLE) {
            synchronized (LazySingle.class) {
                //初次的时候初始化对象
                if (null == SINGLE) {
                    SINGLE = new LazySingle();
                }
            }
        }
        return SINGLE;
    }
}

单例-静态内部类

  • 推荐使用这种方式
  • 静态内部类: jvm提供给我们的同步控制,static 可以进行区块初始化的操作,保证数据在内存是独一份的,final初始化之后,是无法被修改的,所也是线程安全的。(jvm在加载内的时候保证类的同步)
  • 是要我们不使用内部类,就不会再次创建实例对象
  • 这种方式是在性能,和线程安全上更有优化的
public class InnerStaticSingle {

    private InnerStaticSingle() {
    }
    
    public static InnerStaticSingle instance() {
        return SingleHolder.SINGLE;
    }
    //静态内部类
    private static class SingleHolder {
        private static final InnerStaticSingle SINGLE = new InnerStaticSingle();
    }
}

单例-枚举

  • 写法简单,线程安全
  • 不写实例方法的话,默认的情况下,枚举的创建是线程安全的,如果自己添加实例,需要注意线程安全.
  • 注意:在使用成员变量和成员方法的时候,需要保证线程安全
public enum EnumSingle {
    //定义一个枚举的常量就是一个实例
    INSTANCE;
    //在使用成员变量和成员方法的时候,需要保证线程安全
    public void say() {
    }
}

Android 单例实际运用

  • Application
  • 单例引起内存年泄漏
  • eventbus的坑
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容