什么是单例模式?简言之就是确保定义为单例模式的类在程序中有且只有一个实例。单例模式的特色:java
只有一个实例 (只能有一个对象被建立)安全
自我实例化(类构造器私有)多线程
对外提供获取实例的静态方法性能
常见的单例模式实现方式有五种:测试
懒汉式(通常也称之为 饱汉式),具体代码实现以下:线程
public class Singleton { /** * 自我实例化 */ private static Singleton singleton; /** * 构造方法私有 */ private Singleton() { System.out.println("建立单例实例..."); } /** * 对外提供获取实例的静态方法 */ public static Singleton getInstance() { if (null == singleton) { singleton = new Singleton(); } return singleton; } }
从代码实现中能够看到,实例并非在一开始就是初始化的,而是在调用 getInstance()方法后才会产生单例,这种模式延迟初始化实例,但它并不是是线程安全的。code
public class SingleTonTest { /** * 多线程模式下测试懒汉模式是否线程安全 * * @param args */ public static void main(String[] args) { /** * 这里我图方便,直接用Executors建立线程池 * 阿里巴巴开发手册是不推荐这么作的 */ ExecutorService executorService = Executors.newFixedThreadPool(5); for (int i = 0; i < 20; i++) { executorService.execute(() -> System.out.println(Thread.currentThread().getName() + "::" + Singleton.getInstance())); } } }
测试结果截图:对象
懒汉式是在运行时加载对象的,因此加载该单例类时会比较快,可是获取对象会比较慢。且这样作是线程不安全的,若是想要线程安全,能够在getInstance()方法加上synchronized 关键词修饰,但这样会让咱们付出惨重的效率代价。blog
提早建立好实例对象,调用效率高,但没法延时加载,容易产生垃圾,线程安全。排序
public class Singleton { /** * 自我实例化 */ private static Singleton singleton = new Singleton(); /** * 构造方法私有 */ private Singleton() { System.out.println("建立单例实例..."); } /** * 对外提供获取实例的静态方法 */ public static Singleton getInstance() { return singleton; } }
public class Singleton { /** * 自我实例化,volatile修饰,保证线程间可见 */ private volatile static Singleton singleton; /** * 构造方法私有 */ private Singleton() { System.out.println("建立单例实例..."); } /** * 对外提供获取实例的静态方法 */ public static Singleton getInstance() { // 第一次检查,避免没必要要的实例 if (singleton == null) { // 第二次检查,同步,避免产生多线程的问题 synchronized (Singleton.class) { if (singleton == null) { singleton = new Singleton(); } } } return singleton; } }
因为singleton=new Singleton()
对象的建立在JVM中可能会进行重排序,在多线程访问下存在风险,使用volatile
修饰signleton
实例变量,能禁止指令重排,使得对象在多线程间可见,可以有效解决该问题。
双重检查锁定失败的问题并不归咎于 JVM 中的实现 bug,而是归咎于 Java 平台内存模型。内存模型容许所谓的“无序写入”,这也是这些习语失败的一个主要缘由
public class Singleton { /** * 构造方法私有 */ private Singleton() { System.out.println("建立单例实例..."); } private static class SingletonInner { private static Singleton instance = new Singleton(); } private static Singleton getInstance() { return SingletonInner.singleton; } }
这样写充分利用静态内部类的特色——初始化操做和外部类是分开的,只有首次调用getInstance()方法时,虚拟机才加载内部类(SingletonInner.class)并初始化instance, 保证对象的惟一性。
public enum Singleton { INSTANCE }
感受异常简单,默认枚举类建立的对象都是单例的,且支持多线程。
文章首发于本人博客:www.developlee.top, 转载请注明出处!
关注公众号,后台回复666, 领取福利: