[Java] [Singleton] [DCL]

Singleton

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

实现方法

饿汉式singleton

  • 预先加载法
  • class Single {
      private Single() {
        System.out.println("ok");
      }
    
      private static Single instance = new Single();
    
      public static Single getInstance() {
        return instance;
      }
    }
  • 优点:
    • thread safe
    • 调用时速度快(在类加载时已经创建好一个static对象)
  • 缺点:
    • 资源利用率不高(可能系统不需要)
    • 在一些场景下无法使用。比如在single实例的创建依赖参数或配置文件时。

懒汉式singleton

  • 延迟加载法
  • public class LazySingleton {
      private static LazySingleton instance;
      private LazySingleton() {}
    
      public static LazySingleton getInstance() {
        if (instance == null) {
          instance = new LazySingleton();
        }
        return instance;
      }
    }
  • 适用于单线程环境,not trhead-safe,getInstance()方法可能返回两个不同实例。
  • 可以改成thread-safe版本,如下:
    public class LazySingleton {
        private static LazySingleton instance;
        private LazySingleton() {}
    
        public static synchronized LazySingleton getInstance() {
    		if (instance == null) {
    			instance = new LazySingleton();
    		}
            return instance;
        }
    }
  • 优点:不执行getInstance对不会被实例化
  • 缺点:第一次加载时反应不快。每次调用getInstance的同步开销大。(大量不必要的同步)

DCL singleton

  • Double Check Lock
  • 避免每次调用getInstance方法时都同步
  • public class LazySingleton {
      private static LazySingleton instance;
      private LazySingleton() {}
    
      public static LazySingleton getInstance() {
        if (instance == null) {
          synchronized(LazySingleton.class) {
            if (instance == null) {
              instance = new LazySingleton();
            }
          }
        }
        return instance;
      }
    }
  • 第一层判断,避免不必要的同步。第二层判断则是在线程安全的情况下创建实例。
  • 优点:资源利用率高,多线程下效率高。
  • 缺点:第一次加载时反应不快,由于java内存模型一些原因偶尔会失败,在高并发下有一定的缺陷。
  • 上述代码依然存在不安全性:
    instance = new LazySingleton()这条语句实际上不是一个原子操作,它大概包括三件事:
    1. 给LazySingleton的实例分配内存;
    2. 初始化LazySingleton()的构造器;
    3. 将instance对象指向分配的内存空间(在这一步的时候instance变成非null)。

  但是由于Java编译器允许处理器乱序执行(指令重排序),上述2、3点的顺序是无法保证的。(意思是可能instance != null时有可能还未真正初始化构造器)。 
  解决方法是通过将instance定义为volatile的。(volatile有两个语义:1. 保证变量的可见性;2. 禁止对该变量的指令重排序)

  • 参考<<Java并发编程>> P286 ~ P287。在JMM后续版本(>= Java5.0)中,可以通过结合volatile的方式来启动DCL,并且该方式对性能的影响很小。然而,DCL的这种使用方式已经被广泛地抛弃了。
  • (因为volatile屏蔽指令重排序的语义在JDK1.5中才被完全修复,此前的JDK中即使将变量声明为volatile,也仍然不能完全避免重排序所导致的问题,这主要是因为volatile变量前后的代码仍然存在重排序问题。)

static内部类singleton

  • class Single {
      private Single() {}
    
      private static class InstanceHolder {
        private static final Single instance = new Single();
      }
    
      public static Single getInstance() {
        return InstanceHolder.instance();
      }
    }
  • 优点:线程安全,资源利用率高。
  • 缺点:第一次加载时反应不快。
  • 原理:类级内部类(static修饰的成员内部类)只有在第一次使用时才会被加载。

Summary

  • 考虑到效率、安全性等问题,一般常用饿汉式singleton or static内部类singleton。其中后者是常用的singleton实现方法。
时间: 2024-11-05 18:32:19

[Java] [Singleton] [DCL]的相关文章

Java并发——DCL问题

转自:http://www.iteye.com/topic/875420 如果你搜索网上分析dcl为什么在java中失效的原因,都会谈到编译器会做优化云云,我相信大家看到这个一定会觉得很沮丧.很无助,对自己写的程序很没信心.我很理解这种感受,因为我也经历过,这或许是为什么网上一直有人喜欢谈dcl的原因.如果放在java5之前,从编译器的角度去解释dcl也无可厚非,在java5的JMM(内存模型)已经得到很大的修正,如果到现在还只能从编译器的角度去解释dcl,那简直就在污辱java,要知道java

java Singleton

简单介绍下 java-Singleton 恶汉式: 懒汉式: 线程安全: (可被反射破坏单例) 枚举:   <Effective Java>: 单元素的枚举类型是成为实现Singleton的最佳方法.

Java Singleton的3种实现方式

1.通过静态成员字段来实例化 public class Elvis { /** * 通过final的静态成员字段来调用私有的构造函数实例化对象 */ public static final Elvis INSTANCE = new Elvis(); /** * 私有构造函数,防止在类以外被实例化 */ private Elvis() { } public void leaveTheBuilding() { System.out.println("Leave the building...&quo

[翻译] 单例(Singleton)

英文原文: https://sourcemaking.com/design_patterns/singleton 意图 确保一个类只有一个实例,并提供一个访问其实例的全局点: 封装 "即时初始化" (just-in-time initialization)或 "首次使用时初始化" (initialization on first use). 问题 应用需要一个且唯一一个对象的实例.而且,延迟初始化(lazy initialization)和全局访问是必须的. 讨论

java设计模式之单例模式(七种方法)

单例模式:个人认为这个是最简单的一种设计模式,而且也是在我们开发中最常用的一个设计模式. 单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类.我们前面学习的很多类都是单例的.比如最典型的就是Servlet类!Servlet类被设计成单例,被所有线程共享! Java Singleton模式为我们提供了这样实现的可能.使用Singleton的好处还在于可以节省内存,因为它限制了实例的个数,有利于Java垃圾回收(garbage

Top 10 tough core Java interview questions answers programming

Tough core Java interview questions and answersWhat is tough core java interview question ? Why do people look for tough Java questions before going for interview? well I don't thing I need to answer these tough questions because its pretty natural t

设计模式之单例模式Singleton pattern

单例模式Singleton pattern 一种软件设计模式.在核心结构中只包含一个被称为单例的特殊类. 一个类只有一个对象实例,并且自行实例化向整个系统提供. 动机 一个系统中可以存在多个打印任务,但是只能有一个正在工作的任务:一个系统只能有一个窗口管理器或文件系统:一个系统只能有一个计时工具或ID(序号)生成器.如在Windows中就只能打开一个任务管理器.如果不使用机制对窗口对象进行唯一化,将弹出多个窗口,如果这些窗口显示的内容完全一致,则是重复对象,浪费内存资源:如果这些窗口显示的内容不

Java Design Patterns(2)

1.Factory Design pattern 工厂设计模式的优点 (1)工厂设计模式提供了接口而不是实现的代码方法. (2)工厂模式从客户端代码中删除实际实现类的实例化.工厂模式使我们的代码更健壮,耦合更少,易于扩展.例如,我们可以轻松更改PC类实现,因为客户端程序不知道这一点. (3)工厂模式通过继承提供实现和客户端类之间的抽象. JDK中工厂设计模式实列 java.util.Calendar,ResourceBundle和NumberFormat getInstance()方法使用Fac

java设计模型 解析工厂模式、proxy-agent模式、templete模式

1.Factory Design pattern 工厂设计模式的优点 (1)工厂设计模式提供了接口而不是实现的代码方法. (2)工厂模式从客户端代码中删除实际实现类的实例化.工厂模式使我们的代码更健壮,耦合更少,易于扩展.例如,我们可以轻松更改PC类实现,因为客户端程序不知道这一点. (3)工厂模式通过继承提供实现和客户端类之间的抽象. JDK中工厂设计模式实列 java.util.Calendar,ResourceBundle和NumberFormat getInstance()方法使用Fac