文章目录
🐥总结
什么是单例模式?
单例模式属于创建型设计模式,保证一个类仅有一个实例,并提供一个访问它的全局访问点。
单例模式优点
- 在内存里只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例。
- 避免对资源的多重占用。
- 为整个系统提供一个全局可访问的节点。
5种单例模式写法
懒汉式【不推荐】
优点:需要时才会去创建,资源利用率高。
缺点:线程不安全,严格意义上不算是单例模式,使用synchronized关键字去修饰getInstance()方法才是线程安全的,但是效率会变低。
public class Singleton {
private static Singleton INSTANCE;
private Singleton (){}
public static Singleton getInstance() {
if (INSTANCE== null) {
INSTANCE= new Singleton();
}
return INSTANCE;
}
}
饿汉式【可用】
优点:天生线程安全。
缺点:无论是否使用都先创建,占用内存空间。
public class Singleton {
private static Singleton INSTANCE= new Singleton();
private Singleton (){}
public static Singleton getInstance() {
return INSTANCE;
}
}
双重检查锁式(DCL)【推荐使用】
优点:延迟加载,多线程安全,集成了懒汉式和饿汉式的优点,资源利用率高。
缺点:在getInstance()方法中判断了两次null检查,提高了资源利用率,同时添加volatile关键字来修饰Singleton类来禁止指令重排,但是由于JVM虚拟机底层存在乱序执行的问题,有时会失效,所以建议使用内部静态类来代替。
public class Singleton {
private volatile static Singleton INSTANCE;
private Singleton (){
}
public static Singleton getInstance() {
if (INSTANCE== null) {
synchronized (Singleton.class) {
if (INSTANCE== null) {
INSTANCE= new Singleton();
}
}
}
return singleton;
}
}
内部静态类【推荐使用】
优点:延迟加载,线程安全,类加载的时候不会去初始化INSTANCE,直到getInstance()被调用才会去初始化,而且无论调用多少次getInstance(),都是取得同一个INSTANCE。
缺点:无法传参。
public class SingleTon{
private SingleTon(){}
private static class SingleTonHoler{
private static SingleTon INSTANCE = new SingleTon();
}
public static SingleTon getInstance(){
return SingleTonHoler.INSTANCE;
}
}
枚举【推荐使用】
优点:线程安全,防止序列化,在任何情况下都是一个单例。
public enum Singleton {
INSTANCE;
public void whateverMethod() {
}
}
总结
一般情况下,推荐使用DCL、枚举和内部静态类来创建单例模式,饿汉式也可以使用,不推荐使用懒汉式,无论是否线程安全。