单例模式(Singleton Pattern):为什么要用枚举实现

单例模式确保类只有一个实例,并提供全局访问点。饿汉模式在类加载时初始化实例,而懒汉模式在首次调用时创建。为解决懒汉模式的线程安全问题,可以使用双重检查锁,但最佳实践是通过枚举实现单例,这同时避免了反射攻击。

单例模式(Singleton Pattern)

确保一个类只有一个实例,并对外提供统一的访问方式

分类

  1. 饿汉模式:类被加载时就立即初始化并创建唯一实例
  2. 懒汉模式:被客户端首次调用时才创建唯一实例
    1)加入双重检查锁检测机制来保证懒汉模式的线程安全
    2)枚举单例模式来防止反射入侵

饿汉模式

/**
 * 饿汉模式
 */
public class StarvingSingleton {
    /**
     * 类加载时就实例化
     * 私有的构造函数,与类实例
     */
    private static final StarvingSingleton starvingSingleton = new StarvingSingleton();
    private StarvingSingleton(){ }
    /**
     * 获取实例的唯一入口
     * @return  实例
     */
    public static StarvingSingleton getInstance(){
        return starvingSingleton;
    }
}

饿汉模式

public class LazyDoubleCheckSingleton {
    /**
     * volatile 保证其可见性与禁止重排序
     */
    private volatile static LazyDoubleCheckSingleton instance;
    private LazyDoubleCheckSingleton(){}
    /**
     * synchronized:保证其原子性
     * 双重检查锁检测机制:保证线程操作资源过程不被打断
     * @return
     */
    public static LazyDoubleCheckSingleton getInstance(){
        //第一次检测
        if (instance==null){
            //同步
            synchronized (LazyDoubleCheckSingleton.class){
                if (instance == null){
                    /*
                        对象初始化的三个步骤
                        memory = allocate(); //1.分配对象内存空间
                        instance(memory);    //2.初始化对象
                        instance = memory;   //3.设置instance指向刚分配的内存地址,此时instance!=null
                    */
                    instance = new LazyDoubleCheckSingleton();
                }

            }
        }
        return instance;
    }

枚举单例

/**
 * 通过反射可以直接入侵类的私有构造函数,实例化类的对象;
 * 枚举类无法通过反射入侵,因此能够解决上述问题;
 */
public class EnumStarvingSingleton {
    private EnumStarvingSingleton(){}
    public static EnumStarvingSingleton getInstance(){
        return ContainerHolder.HOLDER.instance;
    }
    private enum ContainerHolder{
        HOLDER;
        private EnumStarvingSingleton instance;
        ContainerHolder(){
            instance = new EnumStarvingSingleton();
        } 
    }
}
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值