大话设计模式笔记(十八)の单例模式

单例模式

定义

保证一个类仅有一个实例,并提供一个访问它的全局访问点。

通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象。一个最好的办法就是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以创建,并且它可以提供一个访问该实例的方法。

UML图

方式一:单线程下的单例

/**
 * Created by callmeDevil on 2019/8/17.
 */
public class Singleton {

    private static Singleton instance;

    private Singleton(){} //构造方法私有,防止外界创建实例

    // 获得本类实例的唯一全局访问点
    public static Singleton getInstance(){
        if (instance == null) {
            //若实例不存在,则创建一个新实例,否则直接返回已有实例
            instance = new Singleton();
        }
        return instance;
    }

}

测试

public class Test {
    public static void main(String[] args) {
        Singleton s1 = Singleton.getInstance();
        Singleton s2 = Singleton.getInstance();
        if (s1 == s2) {
            System.out.println("两个对象是相同的实例");
        }
    }
}

测试结果

两个对象是相同的实例

在没有并发问题的情况下,这种方式也是使用比较多的。但缺点也很明显,多线程下根本没法用。

方式二:多线程下的单例

/**
 * Created by callmeDevil on 2019/8/17.
 */
public class SingletonOnLock {

    private static SingletonOnLock instance;

    private SingletonOnLock(){}

    public static SingletonOnLock getInstance(){
        // 同步代码块,只有一个线程能进入,其他阻塞
        synchronized (SingletonOnLock.class){
            if(instance == null){
                instance = new SingletonOnLock();
            }
        }
        return instance;
    }

}

存在问题

当存在对象实例时,完全不用担心并发时导致中创建多个实例,但每次调用 getInstance() 方法时都被加锁,是会影响性能的,因此这个类可以继续改良。

方式三:双重锁定(DCL)

/**
 * Created by callmeDevil on 2019/8/17.
 */
public class SingletonOnDoubleCheckLock {

    private static SingletonOnDoubleCheckLock instance;

    private SingletonOnDoubleCheckLock(){}

    public static SingletonOnDoubleCheckLock getInstance(){
        // 先判断实例是否存在,不存在再考虑并发问题
        if (instance == null) {
            synchronized (SingletonOnDoubleCheckLock.class){
                if(instance == null){
                    instance = new SingletonOnDoubleCheckLock();
                }
            }
        }
        return instance;
    }

}

两次判断实例是否存在的原因

当实例存在时,就直接返回,这是没有问题的。当实例为空并且有两个线程调用 getInstance() 方法时,它们都可以通过第一重 instace == null 的判断,然后由于 synchronized 机制,只有一个线程可以进入,另一个阻塞,必须要在同步代码块中的线程出来后,另一个线程才会进入。而此时如果没有第二重的判断,那第二个线程仍然会创建实例,这就达不到单例的目的了。

但这种方式是最让人“诟病”的一种不推荐方式,技巧看上去很好,但实际上同样影响性能

方式四:静态初始化

/**
 * 该类声明为final ,阻止派生,因为派生可能会增加实例
 * Created by callmeDevil on 2019/8/17.
 */
public final class SingletonStatic {

    // 第一次引用类的任何成员时就创建好实例,同时没有并发问题
    private static final SingletonStatic instance = new SingletonStatic();

    private SingletonStatic(){}

    public static SingletonStatic getInstance(){
        return instance;
    }

}

JVM第一次加载类的时候就已经创建好了实例,如果接下来的很长时间都没有用到的话,占用的内存相当于被浪费了,也不是最让人推荐的一种方式。当然现在的服务器容量也越来越大,单单一个实例的内存也并不是任何情况都要考虑节省。除非追求极致。。

总结

  • 静态初始化的方式是自己被加载时就已经将自己实例化,因此也被称为“饿汉式”。
  • 其他方式是要在第一次被引用时,才会将自己实例化,所以被称为“懒汉式”。
  • 其实单例模式还有很多种实现方式,下面再提一种《大话设计模式》中未提到的实现:静态内部类。 这是楼主在《Java并发编程实战》中看到的,也是最为推荐的一种。

扩展

方式五:静态内部类

/**
 * Created by callmeDevil on 2019/8/17.
 */
public class SingletonStaticClass {

    private SingletonStaticClass() {}

    public SingletonStaticClass getInstande() {
        return InterClass.instance;
    }

    // 静态内部类,没有并发问题
    private static final class InterClass {
        public static SingletonStaticClass instance = new SingletonStaticClass();
    }

}

推荐原因

JVM第一次加载外部的 SingletonStaticClass 时,并不会直接实例化,所以这种方式也属于“懒汉式”。只有在第一次调用 getInstance() 方法时,JVM才会加载内部类 InterClass,接着才实例化静态变量,也就是我们需要的外部类的单例。这样不仅延时了实例化,同时也解决了并发访问的问题,因此该方式是最为推荐的一种方式。

(0)

相关推荐

  • 图解Java设计模式之单例设计模式

    图解Java设计模式之单例设计模式 设计模式介绍 设计模式类型 单例设计模式介绍 饿汉式(静态常量) 饿汉式(静态代码块) 懒汉式(线程不安全) 懒汉式(线程安全,同步方法) 懒汉式(线程安全,同步代 ...

  • 单例模式的实现

    记一下学习单例模式的笔记: 单例就是要保证该类仅有一个实例.实现完全封闭的单例(外部不能new)其实就要两点要求: 全局访问:需要一个该类型的全局静态变量,每次获取实例时都要判断它是否null,不存在 ...

  • 浅谈C++设计模式--单例模式

    单例模式(Singleton)[分类:创建型] [目的:为程序的某个类的访问,提供唯一对象实例] 这估计是设计模式里面最简单的一个类了,下面我们一起看看,单例它是什么?以及如何实现一个单例 基本定义 ...

  • 23种设计模式入门 -- 单例模式

    单例模式:采用一定的方法,使得软件运行中,对于某个类只能存在一个实例对象,并且该类只能提供一个取得实例的方法. 分类: 饿汉式 静态常量方式 静态代码块方式 懒汉式 普通方式,线程不安全 同步方法方式 ...

  • 创建型模式之单例模式

    定义与特点 单例(Singleton)模式的定义:指一个类只有一个实例,且该类能自行创建这个实例的一种模式. 例如,Windows 中只能打开一个任务管理器,这样可以避免因打开多个任务管理器窗口而造成 ...

  • 题画诗与红地毯式的写法(笔记十八) | 明哥

    明哥的图片日志  题画诗与红地毯式的写法 (笔记十八) 一.题画诗的写法 (一)巧妙勾勒,化静为动 所谓题画诗,是指诗人根据画面内容进行高度的概括和提炼,巧妙勾勒,使画面静物形象地再现,变画面的静态为 ...

  • 大话设计模式笔记(十七)の迭代器模式

    迭代器模式 定义 提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该对象的内部表示. 什么时候用? 当你需要访问一个聚集对象,而且不管这些对象是什么都需要遍历的时候,你就应该考虑用迭代器模式. ...

  • 大话设计模式笔记(十三)の状态模式

    举个栗子 问题描述 上班的日子,上午状态好,中午想睡觉,下午渐恢复,加班苦煎熬.根据时间的不同体现不同的工作状态. 简单实现 Work /** * 工作类 * Created by callmeDev ...

  • 大话设计模式笔记(七)の原型模式

    举个栗子 问题描述 要求有一个简历类,必须要有姓名,可以设置性别和年龄,可以设置工作经历,最终需要三份简历. 简单实现 简历类 /** * 简历类 * Created by callmeDevil o ...

  • 大话设计模式笔记(六)の工厂方法模式

    栗子回顾 简单工厂模式: https://www.cnblogs.com/call-me-devil/p/10926633.html 运算类使用工厂方法模式实现 UML图 代码实现 工厂接口 /** ...

  • 大话设计模式笔记(五)の代理模式

    举个栗子 故事是这样的... 一个小伙子喜欢上了隔壁班的一个妹子,但是又不认识,也害羞不好意思主动去说话,于是拜托了同样在这个班的一个朋友去传递自己想要送的礼物... 代码实现 该模式就不上什么简单实 ...

  • 大话设计模式笔记(四)の装饰模式

    举个栗子 问题描述 可以给人搭配嘻哈服或白领装的程序. 简单实现 代码 /** * 人类 * Created by callmeDevil on 2019/6/23. */ public class ...

  • 大话设计模式笔记(二)の策略模式

    举个栗子 问题描述 商场收银软件,营业员根据客户所购买的商品单价和数量,向客户收费. 简单实现 /** * 普通实现 * Created by callmeDevil on 2019/6/1. */ ...

  • 大话设计模式笔记(一)の简单工厂模式

    概要 一个好的程序猿/媛敲出来的代码应该是可维护.可复用.可扩展的,也就是具有较好的灵活性. 为了达到以上目的,在还没敲代码之前,需要事先考虑通过何种方式能够使自己的程序的耦合度降低,最基本的便是面向 ...