单例模式不孤单,8大实例带你吃透单例模式

本文详细介绍了单例模式的概念,包括饿汉式和懒汉式的实现,重点讨论了线程安全与效率的问题。双校验锁和静态内部类的实现提供了线程安全且高效的单例模式解决方案。同时,枚举类型的单例模式可以防止反序列化导致的实例创建。此外,还提到了单例模式在JDK和Spring框架中的应用。
摘要由CSDN通过智能技术生成

概念

单例模式即一个 JVM 内存中只存在一个类的对象实例。

 

分类

1、饿汉式

类加载的时候就创建实例

2、懒汉式

使用的时候才创建实例

当然还有其他的生成单例的方式,双重校验锁,枚举和静态内部类,文中会有介绍。

懒汉式

1) 示例1

public class Singleton {  

    private static Singleton instance;  

    private Singleton (){}  

    public static Singleton getInstance() {  
        if (instance == null) {  
            instance = new Singleton();  
        }  
        return instance;  
    }  

}复制代码

线程不安全,不可用。

2) 示例2

public class Singleton {  

    private static Singleton instance;  

    private Singleton (){}  

    public static synchronized Singleton getInstance() { 
        if (instance == null) {  
            instance = new Singleton();  
        }  
        return instance;  
    }  

}复制代码

同步方法,线程安全,效率低,不推荐。

3) 示例3

public class Singleton {

    private static Singleton singleton;

    private Singleton() {}

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

}复制代码

线程不安全,会产生多个实例,不可用。

饿汉式

无线程安全问题,不能延迟加载,影响系统性能。

4) 示例1

public class Singleton {  

    private static Singleton instance = new Singleton();  

    private Singleton (){}  

    public static Singleton getInstance() {  
        return instance;  
    }  

}复制代码

5) 示例2

public class Singleton {  

    private static Singleton instance = null;  

    static {  
        instance = new Singleton();  
    }  

    private Singleton (){}  

    public static Singleton getInstance() {  
        return instance;  
    }  

}复制代码

6) 双重校验锁

public class Singleton {

    private static volatile Singleton singleton;

    private Singleton() {}

    public static Singleton getInstance() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                if (singleton == null) {
                    singleton = new Singleton();
                }
            }

        }
        return singleton;
    }

}复制代码

双重校验锁,线程安全,推荐使用。

7) 静态内部类

public class Singleton {  

    private static class SingletonHolder {  
        private static final Singleton INSTANCE = new Singleton();  
    }  

    private Singleton (){}  

    public static final Singleton getInstance() {  
        return SingletonHolder.INSTANCE;  
    }  

}复制代码

静态内部类,线程安全,主动调用时才实例化,延迟加载效率高,推荐使用。

8) 枚举

public enum Singleton {  

    INSTANCE;  

    public void whateverMethod() {  

    }  

}复制代码

注意事项

1、考虑多线程问题

2、单例类构造方法要设置为private类型禁止外界new创建

private Singleton() {}

3、如果类可序列化,考虑反序列化生成多个实例问题,解决方案如下

private Object readResolve() throws ObjectStreamException {

    // instead of the object we're on, return the class variable INSTANCE  

    return INSTANCE;  

}复制代码

枚举类型,无线程安全问题,避免反序列华创建新的实例,很少使用。

使用场景

1、工具类对象

2、系统中只能存在一个实例的类

3、创建频繁或又耗时耗资源且又经常用到的对象

下面是单例模式在JDK的应用

 

另外,Spring 容器中的实例默认是单例饿汉式类型的,即容器启动时就实例化 bean 到容器中,当然也可以设置懒汉式 defalut-lazy-init="true" 为延迟实例化,用到时再实例化。

了解更多学习知识

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值