美文网首页
单例的四种模式总结

单例的四种模式总结

作者: 元勰 | 来源:发表于2016-11-06 08:23 被阅读0次

    前言

    单例设计模式在设计模式中的定义是:

    确保一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。

    为了保证内存中只有一个类的对象,我们必须对对象创建加以控制。并能让外部获取对象,我们提供获取本类对象的方法。所以,我们对单例这种类提出如下几个要求:

    • 私有化构造器,将创建本来对象的方法控制在本类中。
    • 提供静态方法供外部获取本类对象。

    但满足以上条件的单例模式代码有很多种,较为常见的有如下几种:

    1. 饿汉式

    2. 懒汉式

    3. Double check模式

    4. InnerClass

    以下我们就对这几种模式依次介绍下。

    饿汉式

    饿汉式的单例是在本类中拥有本类的对象,并且提供了静态方法供外部获取。所以本类中拥有本类对象也必须是静态的。饿汉式主要特点是在类加载时就创建对象:

    public class SingletonFirst {
    
     //本类拥有拥有本类的成员,并且在类一加载的时间就创建
     private static SingletonFirst single=new SingletonFirst();
     //私有化本类的构造器
     public SingletonFirst() {
      super();
     }
     //对外提供获取单例的方法
     public static SingletonFirst getInstance(){
      return single;
     }
    }
    ```
    饿汉式的优点
    + 代码简单,明了,
    
    + 不存在线程安全问题。
    
    缺点:
    + 单例对象生存周期过长,对资源占用比较厉害
    
    ##懒汉式
    为了解决饿汉式的缺点,我们将对象的创建延后到了外部调用静态方法的时候。代码如下:
    ```
    public class SingletonSecond {
    
     // 本类拥有拥有本类的成员,并且在类一加载的时间就创建
     private static SingletonFirst single;
    ​
     // 私有化本类的构造器
     public SingletonSecond() {
      super();
     }
    ​
     // 对外提供获取单例的方法
     public static SingletonFirst getInstance() {
      // 在获取对象时,先行判定有没有。这个对象。有就直接返回
      if (single == null) {
       single = new SingletonFirst();
      }
      return single;
     }
    }
    ```
    饿汉式的优点:
    + 在外部类调用方法的时候才创建对象。减少资源的浪费
    
    缺点:
    + 存在线程安全问题。
    
    ##Double check模式
    double check主要是针对懒汉式的改进。加了同步和双重判定机制。代码如下:
    ```
    public class SingletonDoubleCheck {
    
    ​
     // 内部持有本类成员
     private static SingletonDoubleCheck single;
    ​
     // 私有化构造器
     private SingletonDoubleCheck() {
      super();
     }
    ​
     // 对外提供获取本类对象的方法
     public static SingletonDoubleCheck getInstance() {
      // 外部的判断,用来消除同步代码块在线程创建以后资源消耗过多的问题
      if (single == null) {
       // 用同步代码块来确保在多线程下懒汉式的线程安全问题
       synchronized (SingletonDoubleCheck.class) {
        // 检测对象是否存在。不存在则创建对象
        if (single == null) {
         single = new SingletonDoubleCheck();
        }
       }
      }
      return single;
     }
    }
    ```
    double check模式的优点:
    + 不存在线程安全问题。
    
    + 保证了单例在需要时加载
    
    缺点:
    + 相对于前面两种模式代码比较难以理解。
    
    InnerClass
    利用静态内部类的**随着调用而加载**的特性,在静态内部类中创建本类的静态成员变量。因为静态内部来唯一,而保重外部类对象的唯一性。
    ```
    public class SingletonInner {
    
     // 私有化构造器
     private SingletonInner() {
      super();
     }
    ​
     // 提供静态内部类用于创建单例对象
     private static class InnerClass {
      // 在内部类内部创建外部对象。并且设置为静态以保证在内部类中对象唯一。
      private static SingletonInner single = new SingletonInner();
     }
    ​
     // 提供静态方法,对外提供访问接口
     public static SingletonInner getInstance() {
      return InnerClass.single;
     }
    }
    ```
    InnerClass方式解决了单例的生存周期和线程安全问题。并且代码也简单明了。
    ##总结
    以上几种叫常见的几种单例模式。懒汉式和饿汉式最容易理解,但使用时都存在一些问题。故我们都不用,我们主要使用下面这两种模式来解决实际工作中的问题。

    相关文章

      网友评论

          本文标题:单例的四种模式总结

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