美文网首页设计模式
(一)设计模式 单例模式

(一)设计模式 单例模式

作者: 科技猿人 | 来源:发表于2021-04-25 21:37 被阅读0次

设计模式代表了最佳的实践

引言

  • 创建型模式。
  • 主要特点:简单,样式多。
  • 主要解决:一个全局使用的类频繁地创建与销毁。
  • 关键代码:构造函数是私有的。

说明

  • 单例只能有一个实例。
  • 单例类必须自己创建自己的唯一实例。
  • 单例类必须给所有其他对象提供这一实例。

优缺点

  • 优点:内存里只有一个实例,减少了内存的开销(频繁创建)。
  • 缺点:没有接口,不能继承,与单一职责原则冲突。

实现方式

饿汉
public class Singleton {
    private static final Singleton instance = new Singleton();

    private Singleton() {
    }

    public static Singleton getInstance() {
        return instance;
    }
}

恶汉模式:没有加锁,执行效率高。但是类加载时就初始化,浪费内存,容易产生垃圾对象。

懒汉
public class Singleton {
    private static Singleton instance;

    private Singleton() {
    }

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

懒汉模式:第一次调用才初始化,解决了恶汉模式的内存问题。但是每次获取实例都需要同步,加锁影响效率。

双重检查
public class Singleton {
    private volatile static Singleton singleton;

    private Singleton() {
    }

    public static Singleton getSingleton() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                if (singleton == null) {
                    singleton = new Singleton();
                }
            }
        }
        return singleton;
    }
}

双重检查模式

  • 第一次判空:在已经实例化的情况下(99.99%的情况),省去了同步操作。
  • synchronized:在首次实例化时,保证线程同步。
  • 第二次判空:在首次实例化遇到多线程问题时(锁生效的场景),完成实例化后的线程执行不再次进行实例化处理。
  • volatile:保证并发场景的有序性。

为什么要保证有序性?

 对象创建的执行步骤如下:

  • 正常情况
    • 1.分配空间
    • 2.初始化
    • 3.引用赋值
  • 重排情况(JIT优化)
    • 1.分配空间
    • 2.引用赋值
    • 3.初始化

 在单线程模型的场景下,指令重排不会影响执行结果。JIT指令重排的优化,并不会考虑并发场景。
 从上面的重排时序上看,就发现了为什么要使用volatile来保证有序性了。因为引用赋值在前,初始化在后,会导致singleton引用已经不为空了,但是还没有初始化,调用线程会拿到一个没有初始化的引用进行方法调用,会异常奔溃。

ps:如果想查看对象的创建执行步骤,可以使用以下命令:

  • 生成字节码文件:javac XX.java;
  • 对生成的字节码文件反汇编:javap -c -v XX.class;
  • 通过汇编指令以及常量池序号,对应进行查找即可。
静态内部类
public class Singleton {
    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }

    private Singleton() {
    }

    public static final Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}

静态内部类模式:实现更简单。

枚举
public enum Singleton {
    INSTANCE;

    public void todo() {
    }
}

枚举模式:实现单例模式的最佳方法。更简洁,自动支持序列化机制,防止反序列化。缺点就是可读性差(这玩意儿咋看都不像单例)。

实现方式总结

单例模式实现方案对比

小编的博客系列

设计模式 全家桶

相关文章

  • 单例模式Java篇

    单例设计模式- 饿汉式 单例设计模式 - 懒汉式 单例设计模式 - 懒汉式 - 多线程并发 单例设计模式 - 懒汉...

  • 设计模式 - 单例模式

    设计模式 - 单例模式 什么是单例模式 单例模式属于创建型模式,是设计模式中比较简单的模式。在单例模式中,单一的类...

  • 2018-04-08php实战设计模式

    一、单例模式 单例模式是最经典的设计模式之一,到底什么是单例?单例模式适用场景是什么?单例模式如何设计?php中单...

  • python中OOP的单例

    目录 单例设计模式 __new__ 方法 Python 中的单例 01. 单例设计模式 设计模式设计模式 是 前人...

  • 单例

    目标 单例设计模式 __new__ 方法 Python 中的单例 01. 单例设计模式 设计模式设计模式 是 前人...

  • 单例模式

    JAVA设计模式之单例模式 十种常用的设计模式 概念: java中单例模式是一种常见的设计模式,单例模式的写法...

  • 设计模式

    常用的设计模式有,单例设计模式、观察者设计模式、工厂设计模式、装饰设计模式、代理设计模式,模板设计模式等等。 单例...

  • 设计模式之单例模式

    单例设计模式全解析 在学习设计模式时,单例设计模式应该是学习的第一个设计模式,单例设计模式也是“公认”最简单的设计...

  • python 单例

    仅用学习参考 目标 单例设计模式 __new__ 方法 Python 中的单例 01. 单例设计模式 设计模式设计...

  • 基础设计模式:单例模式+工厂模式+注册树模式

    基础设计模式:单例模式+工厂模式+注册树模式 单例模式: 通过提供自身共享实例的访问,单例设计模式用于限制特定对象...

网友评论

    本文标题:(一)设计模式 单例模式

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