随笔-设计模式-单例模式

单例模式

所谓单例,即为仅此一例。在整个系统中,一个类只有一个实例对象,并且该类只提供一个取得实例的静态方法。
八种单例模式:

  • 饿汉式(静态常量)
  • 饿汉式(静态代码块)
  • 懒汉式(线程不安全)
  • 懒汉式(同步方法,线程安全)
  • 懒汉式(同步代码块,线程安全)
  • 双重检查
  • 静态内部类
  • 枚举

一、饿汉式(静态常量)

  • 类的内部创建私有对象
  • 构造器私有化
  • 对外暴露一个静态方法,用来获取实例
public class SingleTon {
  private static final SingleTon SINGLE_TON = new SingleTon();
  public static SingleTon getInstance() {
    return SINGLE_TON;
  }
  private SingleTon() {
  }
}

优点:简单,利用了类加载机制,在类加载时候完成了实例化,避免了线程同步问题。
缺点:没有达到Lazy Loading的效果。如果始终没有使用这个实例,那么就会造成内存浪费。

二、饿汉式(静态代码块)

跟上面差不多,也是在类加载时候完成实例化。

public class SingleTon {
  private static final SingleTon SINGLE_TON;
  static {
    SINGLE_TON = new SingleTon();
  }
  public static SingleTon getInstance() {
    return SINGLE_TON;
  }
  private SingleTon() {
  }
}

优缺点同上。

三、懒汉式(线程不安全)

  • 构造器私有化
  • 对外暴露一个静态方法,用来获取实例
  • 在调用静态方法时候才创建实例
public class SingleTon {
  private static SingleTon SINGLE_TON;

  private SingleTon() {
  }

  public static SingleTon getInstance() {
    if (SINGLE_TON == null) {
      SINGLE_TON = new SingleTon();
    }
    return SINGLE_TON;
  }
}

优点:有Lazy Loading效果
缺点:线程不安全,只能在单线程下使用

四、懒汉式(同步方法,线程安全)

在上面基础加上同步锁

public class SingleTon {
  private static SingleTon SINGLE_TON;

  private SingleTon() {
  }

  public static synchronized SingleTon getInstance() {
    if (SINGLE_TON == null) {
      SINGLE_TON = new SingleTon();
    }
    return SINGLE_TON;
  }
}

优点:有Lazy Loading效果,并且线程安全
缺点:因为有同步锁,效率低

五、懒汉式(同步代码块,线程安全)

跟上面差不多,只是缩小了同步范围

public class SingleTon {
  private static SingleTon SINGLE_TON;

  private SingleTon() {
  }

  public static SingleTon getInstance() {
    if (SINGLE_TON == null) {
      synchronized (SingleTon.class) {
        SINGLE_TON = new SingleTon();
      }
    }
    return SINGLE_TON;
  }
}

优缺点同上

六、双重检查

在懒汉式(同步代码块,线程安全)基础上优化

public class SingleTon {
  private volatile static SingleTon SINGLE_TON;

  private SingleTon() {
  }

  public static SingleTon getInstance() {
    if (SINGLE_TON == null) {
      synchronized (SingleTon.class) {
        if (SINGLE_TON == null) {
          SINGLE_TON = new SingleTon();
        }
      }
    }
    return SINGLE_TON;
  }
}

优点:利用了Double-Check,既保证了线程安全,又减少了同步代码块的执行,还实现了Lazy Loading
缺点:一开始,会有部分线程会执行两次判断(影响可以忽略)

七、静态内部类

  • 构造器私有化
  • 写一个静态内部类,该类中有一个静态属性SingleTon
  • 对外暴露一个静态方法,用来获取实例
public class SingleTon {
  private SingleTon() {
  }

  private static class InnerSingleTon {
    private static final SingleTon SINGLE_TON = new SingleTon();
  }

  public static SingleTon getInstance() {
    return InnerSingleTon.SINGLE_TON;
  }
}

优点:利用类加载机制(静态内部类在类加载时候不会实例化,在被调用时候才会实例化),实现Lazy Loading,线程安全,效率高

八、枚举

public enum SingleTon {
  SINGLE_TON;

  public void doSome() {
    System.out.println("ok");
  }

}

优点:线程安全,防止反序列化重新创建新的对象

总结

双重检查、 静态内部类、枚举都推荐使用,个人用最后两种比较多

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值