美文网首页Android高级进阶知识
单例模式的几种可行方式

单例模式的几种可行方式

作者: reggie1996 | 来源:发表于2018-08-06 23:42 被阅读19次
    单例的概念

      单例模式是一种对象创建模式,他用于产生一个对象的具体实例,它可以确保系统中一个类只产生一个实例。
      对于频繁使用的对象,可以省略创建对象所花费的时间,对于那些重量级的对象而言,是非常可观的一笔系统开销。由于 new 操作的次数减少,因而对系统内存的使用频率也会降低,这将减轻 GC 压力,缩短 GC 时间。
      单例模式的核心在于通过一个接口返回唯一的对象实例。单例主要需要解决的问题就是要把创建对象的权限收回来,比如说单例最重要的方法就是把类的构造方法设为 private,这就是把创建对象的权限收回来,而让类自身来完成创建自己实例的工作,然后让类提供外部可以访问这个实例的方法。
      下面是几种可选择的单例实现方式。

    饿汉式:
    /**
     * 饿汉式
     */
    public class Singleton1 {
    
        private static Singleton1 instance = new Singleton1();
    
        private Singleton1(){
    
        }
    
        public static Singleton1 getInstance() {
            return instance;
        }
    
        public void method(){
            // do something.
        }
    
    }
    

      饿汉模式是在类创建时就创建实例,是线程安全的。所谓饿汉,就是因为在类的实例还未被需要使用时,这个实例就被创建了,因此是不能懒加载,这是饿汉式的缺点。

    懒汉DCL式:
    /**
     * 懒汉DCL式
     */
    public class Singleton2 {
    
        private static volatile Singleton2 instance;
    
        private Singleton2(){
    
        }
    
        public static Singleton2 getInstance(){
            if (instance == null){
                synchronized (Singleton2.class){
                    if (instance == null){
                        instance = new Singleton2();
                    }
                }
            }
            return instance;
        }
    
        public void method(){
            // do something.
        }
    
    }
    

      懒汉模式,相对于饿汉模式区别,是懒汉模式的实例在类的实例需要使用时,才会被创建,是懒加载。在此之上,我们这里又用到了 DCL(double checked locking)的方式,来保证单例的线程安全,还用了 volatile 关键字来禁止 JVM 对 instance 相关的操作的指令重排序优化,保证 instance 是唯一的。

    静态内部类式:
    /**
     * 静态内部类式
     */
    public class Singleton3 {
    
        private Singleton3(){
    
        }
    
        public static Singleton3 getInstance(){
            return InstanceHolder.instance;
        }
    
        public void method(){
            // do something.
        }
    
        /**
         * 持有单例的静态内部类
         */
        private static class InstanceHolder{
            static final Singleton3 instance = new Singleton3();
        }
    
    }
    

      静态内部类单例是懒加载的。静态内部类实现单例主要是依靠 JVM 提供给我们的同步控制,利用 static 和 final 两个关键字, 通过 static 进行区块初始化数据,保证数据在内存中是独一份的,final 字段保证数据初始化后无法被修改。由此来保证单例的同步控制。
      静态内部类单例相对于饿汉DCL式的优点是,由于没用使用 synchronized,所以性能上会有一些优势。

    枚举式:
    /**
     * 枚举式
     */
    public enum Singleton4 {
    
        INSTANCE;
    
        public void method(){
            // do something.
        }
    
    }
    

      枚举式是利用 enum 的特性来保证其唯一性,它的优点显而易见,实现起来十分简单,同时它又是是线程安全的。

    分别的使用方法:
            //饿汉式
            Singleton1.getInstance().method();
            //懒汉DCL式
            Singleton2.getInstance().method();
            //静态内部类式
            Singleton3.getInstance().method();
            //枚举式
            Singleton4.INSTANCE.method();
    

      以上是单例模式的几种可行方式。

    相关文章

      网友评论

        本文标题:单例模式的几种可行方式

        本文链接:https://www.haomeiwen.com/subject/zhfivftx.html