美文网首页
单利设计模式

单利设计模式

作者: 往事一块六毛八 | 来源:发表于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();
    }
}

相关文章

  • JavaJavascript基础进阶(十七)JS中常用的设计模式

    单利设计模式、构造原型设计模式、发布订阅设计模式、promise设计模式 单利模式 构造原型设计模式 最贴近OOP...

  • 细品 javascript 设计模式(单利模式)

    我尽量用最少的文字,最少的篇幅,讲明白设计模式的方方面面。文章连接 理解单利模式 上代码:通用的惰性单利模式 单利...

  • 单利设计模式

    单利模式的介绍 单利模式是应用最广的模式之一,也可能是很多初级工程师唯一会使用的设计模式。在应用这个模式时,单利对...

  • 2018-05-14

    单利设计模式 懒汉式 单例模式 饿汉式 单利模式 懒汉式与饿汉式的区别: 双重锁式 单例模式 (DCL )

  • 设计模式---单利

    public class Text02 {public static void main(String[] arg...

  • 单利设计模式(二)

    单利模式的问题 其实在之前就写了一篇关于单利设计模式中的懒汉式,也解决了懒汉式中存在的多线程安全问题,但是深入理解...

  • 单利模式设计详解

    @interfaceSingleton:NSObject + (Singleton *)sharedSinglet...

  • 设计模式

    设计模式的分类 创建型模式(5种) 工厂方法模式、抽象工厂模式、单利模式、构建者模式、原型模式。 结构型模式(7种...

  • 设计模式  ,JVM,数据库

    三、设计模式 1、简述一下你了解的设计模式。 2、写出单利模式,懒汉和饿汉 四、JVM 1、描述一下JVM加载cl...

  • 1,设计模式-策略模式

    设计模式分类1,创建型模式,共5种:工厂方法模式,抽象工厂模式,单利模式,建造者模式,原型模式。2,结构型模式,共...

网友评论

      本文标题:单利设计模式

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