美文网首页
单利设计模式

单利设计模式

作者: 往事一块六毛八 | 来源:发表于2019-11-30 22:40 被阅读0次

    单利模式的介绍

    单利模式是应用最广的模式之一,也可能是很多初级工程师唯一会使用的设计模式。在应用这个模式时,单利对象的类必须保证只有一个实例存在。许多情况下整个系统只需要拥有一个全局对象,这样有利于我们协调系统整体的行为,降低支援的消耗。

    定义

    确保只有一个实例,而且自行实例化并向这个系统提供这个实例。

    单例模式使用场景(套路)

    确保某个类有且只有一个对象的场景,避免产生多个对象消耗过多的资源,或者某种类型的对象应该有且只有一个。例如,创建一个对象需要消耗的资源过多,如访问IO和数据库等资源,这时候就要考虑使用单利模式。

    单例模式UML类图

    单利模式.png

    角色介绍:
    (1)Client----高层客户端;
    (2)Singleton----单利类
    实现单例模式主要有如下几个关键点:

    • 构造函数不对外开放,一般为private
    • 通过一个静态方法或者枚举返回单利类对象
    • 确保单利类的对象有且只有一个,尤其是在多线程环境下
    • 确保单利类对象在反序列化是不会重新构建对象

    注意:
    通过将单利类的构造函数私有化,使得客户端代码不能通过new的形式手动构造单利类的对象。单利类会暴露一个共有静态方法,客户端需要调用这个静态方法获取到单利类的唯一对象,在获取这个单利对象的过程中需要确保线程安全,即在多线程环境下构造单利类的对象也是有且只有一个。

    单利类的几种实现方式

    1:饿汉式

    public class SingleTon {
        public static final SingleTon mInstance = new SingleTon();
    
        //构造函数私有
        private SingleTon() {
        }
        //公有的静态函数,对外暴露获取单利对象的接口
        public static SingleTon getInstance(){
            return mInstance;
        }
    }
    

    从上述代码可知,SingleTon 类不能通过new的形式构造对象,只能通过
    SingleTon.getInstance()函数获取,而这个对象时静态对象,并且在声明的时候就已经初始化,这就保证了SingleTon对象的唯一性。

    2:懒汉式
    懒汉式是声明一个静态对象,并且在用户第一次调用getInstance()时进行初始化

    public class SingleTon {
        public static SingleTon mInstance ;
    
        //构造函数私有
        private SingleTon() {
        }
        //公有的静态函数,对外暴露获取单利对象的接口
        public static synchronized SingleTon getInstance(){
            if (mInstance==null){
                mInstance=new SingleTon();
            }
            return mInstance;
        }
    }
    

    3:Double Check Lock(DCL)实现单利

    public class SingleTon {
        public static SingleTon mInstance;
    
        //构造函数私有
        private SingleTon() {
        }
    
        //公有的静态函数,对外暴露获取单利对象的接口
        public static synchronized SingleTon getInstance() {
            if (mInstance == null) {
                synchronized (SingleTon.class) {
                    if (mInstance == null) {
                        mInstance = new SingleTon();
                    }
                }
            }
            return mInstance;
        }
    }
    

    双重锁的亮点在于双重判断,相对于懒汉式的写法,减少了没必要的同步判断。

    • 第一层判断主要是避免不必要的同步
    • 第二层判断是为了在null的情况下创建实例

    那么DCL模式就没有缺点?
    假设线程A执行到mInstance = new SingleTon();语句,这里看起来是一句代码,但实际上它并不是一个原子操作,这段代码最终会被变异成多条汇编指令,大致做了三件事:
    (1)给mInstance 实例分配内存;
    (2)调用Singleton()的构造函数,初始化成员字段
    (3)将mInstace对象指向分配的内存空间(此时mInstace就不是null了)
    如果指令按照顺序执行倒也无妨,但在jdk1.5之前,JVM为了优化指令,提高程序运行效率,允许指令重排序上面的执行顺序有可能是这样的
    (a)给mInstance 实例分配内存;
    (b)将mInstace对象指向分配的内存空间(此时mInstace就不是null了)
    (c)调用Singleton()的构造函数,初始化成员字段
    如果是后者,并且执行完了b操作(c操作之前),被切换到线程B,这时候mInstance 因为已经在线程A执行了b操作,mInstance 不为空,所以线程B直接取走mInstance ,在使用时就会出错,这就是DCL失效问题。
    具体来说就是synchronized虽然保证了线程的原子性(即synchronized块中的语句要么全部执行,要么一条也不执行),但单条语句编译后形成的指令并不是一个原子操作(即可能该条语句的部分指令未得到执行,就被切换到另一个线程了)。
    根据以上分析可知,解决这个问题的方法是:禁止指令重排序优化,即使用volatile变量。

    public class SingleTon {
        public volatile static SingleTon mInstance;
    
        //构造函数私有
        private SingleTon() {
        }
    
        //公有的静态函数,对外暴露获取单利对象的接口
        public static synchronized SingleTon getInstance() {
            if (mInstance == null) {
                synchronized (SingleTon.class) {
                    if (mInstance == null) {
                        mInstance = new SingleTon();
                    }
                }
            }
            return mInstance;
        }
    }
    
    

    4:静态内部类单利模式

    public class SingleTon {
        public volatile static SingleTon mInstance;
    
        //构造函数私有
        private SingleTon() {
        }
    
        public static SingleTon getInstance() {
            return SingletonHolder.mInstance;
        }
    
        /**
         * 静态内部类
         */
        private static class SingletonHolder {
            private static final SingleTon mInstance = new SingleTon();
        }
    }
    

    当第一次加在Singleton类时并不会初始化mInstace,只有在第一次调用
    SingleTon.getInstance()方法时才会导致mInstace被初始化。因此,第一次调用getInstace()方法会导致虚拟机加在SingletonHolder类。这种方式不仅能够确保线程安全,也能保证单利对象的唯一性,同时也延迟了单利的实例化,所以这是土建使用的单利模式实现方式。

    5:枚举单利

    public enum  SingleTon {
        INSTANCE;
        public void doSomething(){
            System.out.println("do sth...");
        }
    }
    
    

    有点:

    • 实现方式简单
    • 默认情况下枚举实例的创建是线程安全的,并且在任何情况下它都是一个单利。

    6:使用容器实现单利模式

    public class SingletonManager {
        private static Map<String, Object> objectMap = new HashMap<>();
    
        private SingletonManager() {
        }
    
        public static void registerServiec(String key, Object instance) {
            if (!objectMap.containsKey(key)) {
                objectMap.put(key, instance);
            }
        }
    
        public static Object getService(String key) {
            return objectMap.get(key);
        }
    }
    

    在程序的初始,将多种单利类型注入到一个统一的管理类中,在使用时根据key获取对象对应类型的对象。这种方式使得我们可以管理多种类型的单利,并且在使用时可以通过统一的接口进行后去操作,降低用户的使用成本,也对用户隐藏了具体实现,降低了耦合度。

    单利模式的实例--Activity的管理

    
    public class ActivityManager {
        private static volatile ActivityManager mInstance;
        // 集合用谁 List LinkedList Stack  ?? 删除和添加比较多
        private Stack<Activity> mActivities;
    
        private ActivityManager(){
            mActivities = new Stack<>();
        }
    
        // 虽说解决了线程安全的问题,但是又会出现效率的问题,
        // 即保证线程的安全同是效率也是比较高的
        // 这种方式其实还是会有问题?
        public static ActivityManager getInstance() {
            if (mInstance == null) {
                synchronized (ActivityManager.class) {
                    if (mInstance == null) {
                        mInstance = new ActivityManager();
                    }
                }
            }
            return mInstance;
        }
    
        /**
         * 添加统一管理
         * @param activity
         */
        public void attach(Activity activity){
            mActivities.add(activity);
        }
    
        /**
         * 移除解绑 - 防止内存泄漏
         * @param detachActivity
         */
        public void detach(Activity detachActivity){
            // for 去移除有没有问题? 一边循环一边移除会出问题 ,
            // 既然这个写法有问题,自己又想不到什么解决方法,参考一下别人怎么写的
            /*for (Activity activity : mActivities) {
                if(activity == detachActivity){
                    mActivities.remove(activity);
                }
            }*/
            int size = mActivities.size();
            for (int i = 0; i < size; i++) {
                Activity activity = mActivities.get(i);
                if (activity == detachActivity) {
                    mActivities.remove(i);
                    i--;
                    size--;
                }
            }
        }
    
        /**
         * 关闭当前的 Activity
         * @param finishActivity
         */
        public void finish(Activity finishActivity){
            // for 去移除有没有问题?
            /*for (Activity activity : mActivities) {
                if(activity == finishActivity){
                    mActivities.remove(activity);
                    activity.finish();
                }
            }
    */
            int size = mActivities.size();
            for (int i = 0; i < size; i++) {
                Activity activity = mActivities.get(i);
                if (activity == finishActivity) {
                    mActivities.remove(i);
                    activity.finish();
                    i--;
                    size--;
                }
            }
        }
    
        /**
         * 根据Activity的类名关闭 Activity
         */
        public void finish(Class<? extends Activity> activityClass){
            // for 去移除有没有问题?
            /*for (Activity activity : mActivities) {
                if(activity.getClass().getCanonicalName().equals(activityClass.getCanonicalName())){
                    mActivities.remove(activity);
                    activity.finish();
                }
            }*/
    
            int size = mActivities.size();
            for (int i = 0; i < size; i++) {
                Activity activity = mActivities.get(i);
                if (activity.getClass().getCanonicalName().equals(activityClass.getCanonicalName())) {
                    mActivities.remove(i);
                    activity.finish();
                    i--;
                    size--;
                }
            }
        }
    
        /**
         * 退出整个应用
         */
        public void exitApplication(){
    
        }
    
        /**
         * 获取当前的Activity(最前面)
         * @return
         */
        public Activity currentActivity(){
            return mActivities.lastElement();
        }
    }
    

    相关文章

      网友评论

          本文标题:单利设计模式

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