Java设计模式之单例模式

定义

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

单例的使用场景

在一个系统中,要求一个类有且仅有一个对象,具体使用场景如下:

  • 整个项目需要一个共享访问点或共享数据。
  • 创建一个对象需要耗费的资源过多,比如访问 I/O或者数据库等资源。
  • 工具类对象。

单例模式的6六种写法

1. 饿汉模式

public class Singleton {
    private static Singleton instance = new Singleton();
    private Singleton() {
    }
    public static Singleton getInstance(){
        return instance;
    }
}

优点:

  • 这种方式在类加载时就完成初始化了,获取对象但速度快。
  • 避免多线程但同步问题。

缺点:

  • 类加载较慢。
  • 没有达到懒加载的效果,如果从始至终都未使用果这个实例,这会造成内存的浪费。

2. 懒汉模式(线程不安全)

public class Singleton {
    private static Singleton instance;
    private Singleton() {
    }
    public static Singleton getInstance(){
        if(instance == null){
            instance = new Singleton();
        }
        return instance;
    }
}

优点:

  • 节省资源

缺点:

  • 第一次实例化对象时较慢。
  • 多线程时不能正常工作。

3. 懒汉模式(线程安全)

public class Singleton {
    private static Singleton instance;
    private Singleton() {
    }
    public static synchronized Singleton getInstance(){
        if(instance == null){
            instance = new Singleton();
        }
        return instance;
    }
}

优点:

  • 多线程中很好的工作。

缺点:

  • 每次调用 getInstance 方法时都需要同步,这会造成不必要的同步开销。

大部分时候我们是用不到同步的,所以,不建议用这种模式。


4. 双重检查模式

public class Singleton {
    private volatile static Singleton instance;
    private Singleton() {
    }
    public static  Singleton getInstance(){
        if(instance == null){
            synchronized (Singleton.class) {
                if(instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

优点:

  • 资源利用率高。
  • 避免多余同步。
  • 线程安全。

缺点:

  • 使用了 volatile 或多或少会影响性能。
  • 第一次实例化对象时较慢。
  • 在高并发某些情况下会出现失效问题。

DCL (Double Check Lock) 在高并发环境下也会有一定的缺陷,DCL 虽然在一定程度上解决了资源的消耗、多余的同步、线程安全问题等问题,但还是在某些情况会出现失效的问题,也就是 DCL 失效。这里建议用静态内部类单例模式来代替 DCL


5. 静态内部类单例模式 (推荐使用)

public class Singleton {
    private Singleton() {
    }
    public static  Singleton getInstance(){
        return SingletonHolder.sInstance;
    }
    private static class SingletonHolder {
        private static final Singleton sInstance = new Singleton();
    }
}

第一次加载 Singleton 类时并不会初始化 sInstance。 只有在第一次调用 getInstance 方法时虚拟机才会加载 SingletonHolder 并初始化 sInstance。

优点:

  • 资源利用率高。
  • 避免多余同步。
  • 线程安全。

缺点:

  • 暂无

6. 枚举单例

public enum Singleton {
    INSTANCE;
    public void doSomeThing() {
        
    }
}

优点:

  • 任何情况下都是单例。
  • 默认线程安全。

缺点:

  • 简单
  • 可读性不高。

总结:到这里6种单例写法已介绍完,至于选择那种形势的单例模式,取决与你项目本身情况:是否复杂的高并发环境,或者是否需要控制单例对象的资源消耗。

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

推荐阅读更多精彩内容

  • 单例模式(SingletonPattern)一般被认为是最简单、最易理解的设计模式,也因为它的简洁易懂,是项目中最...
    成热了阅读 9,787评论 4 34
  • 1.单例模式概述 (1)引言 单例模式是应用最广的模式之一,也是23种设计模式中最基本的一个。本文旨在总结通过Ja...
    曹丰斌阅读 8,064评论 6 47
  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,859评论 8 265
  • 参加线下活动到底为何 从2017年以后才了解有meetup这种活动,就是一群人找个时间线下聚一聚。 接着参加了许许...
    林春龙阅读 1,560评论 0 1
  • 以前觉得幸福是一个空洞的词,空洞到不知道现实生活怎么实现,是的,好像现实生活中没有岁月静好,有的是一地鸡毛,但是慢...
    你好3306阅读 936评论 0 0