单例模式大全

4 篇文章 0 订阅

饿汉式

// 饿汉式单例
public class Hungry {
    // 可能会浪费空间
    private byte[] data1 = new byte[1024*1024];
    private byte[] data2 = new byte[1024*1024];
    private byte[] data3 = new byte[1024*1024];
    private byte[] data4 = new byte[1024*1024];
    private Hungry(){
    }
    private final static Hungry HUNGRY = new Hungry();
    public static Hungry getInstance(){
        return HUNGRY;
    }
}

优点: 无同步问题, 类加载时就完成初始化。

缺点:提前加载,导致性能浪费。 实现延迟加载最好。

         无法避免通过 反射 拿到构造方法,去创建新的实例。

懒汉式(DCL)

双重锁检查加volatile(锁内外各检查一次)

public class LazyMan {
    private LazyMan(){ }
    // 双重锁的判断,在两个线程中是不可见的,一个更改,另一个可能看不到。这样外锁才有意义 
    private volatile static LazyMan lazyMan;
    public static LazyMan getInstance(){
        //外锁是为了避免不必要的等待,已经有锁了 ,就不需要在进去。
        if (lazyMan==null){
            synchronized (LazyMan.class){
                // 为了保证实例 只有一个
                if (lazyMan==null){
                    lazyMan = new LazyMan(); // 不是一个原子性操作
                }
            }
        }
        return lazyMan;
    }
}

优点: 延迟加载, 解决同步问题。

缺点:无法避免反射突破。

volatile :双重锁的判断,在两个线程中是不可见的,一个更改,另一个可能看不到。这样外锁才有意义...

                强制线程使用该变量时,从主存获取新的值,而不是直接用线程内的值。线程内的值可能是空,但是线程外可能已经被修改过

            volatile :1保证可见性,要求每次都从主存获取新值,2,不保证原子性 ,2,禁止指令重排

         这里主要是基于指令重排的的影响,假设线程a,b争抢, a先获取锁,创建对象,但是创建对象不是原子的,有几个过程,由于指令重排可能导致,a先创建了指向内存的引用,但是内存内并没有做好初始化。而此时b获取Cpu执行权,外层锁判断对象就不为空,从而获取到一个不完整的实例,造成错误

外锁 :外锁是为了避免不必要的等待,已经有锁了 ,就不需要在进去。

内锁:为了保证实例 只有一个

反射破解

    public static void main(String[] args) throws Exception {
        Constructor<LazyMan> declaredConstructor = LazyMan.class.getDeclaredConstructor(null);
        LazyMan instance = declaredConstructor.newInstance();
        LazyMan instance2 = declaredConstructor.newInstance();
        System.out.println(instance);
        System.out.println(instance2);
    }

DCL+ 加密构造

public class LazyMan {
    private static String qinjiang = "asdadasdad";
    private LazyMan(){
        synchronized (LazyMan.class){
            if (qinjiang == "asdadasdad"){
                qinjiang = "";
            }else {
                throw new RuntimeException("不要试图使用反射破坏异常");
            }
        }
    }
    private volatile static LazyMan lazyMan;
    // 双重检测锁模式的 懒汉式单例  DCL懒汉式
    public static LazyMan getInstance(){
        if (lazyMan==null){
            synchronized (LazyMan.class){
                if (lazyMan==null){
                    lazyMan = new LazyMan(); // 不是一个原子性操作
                }
            }
        }
        return lazyMan;
    }
}

由于类加载所以,第一个实例的构造,必然会把字符串置空,此时即使通过反射,也只能拿到空的属性。

内部类

public class Holder {
    private Holder(){

    }
    public static Holder getInstace(){
        return InnerClass.HOLDER;
    }

    public static class InnerClass{
        private static final Holder HOLDER = new Holder();
    }
}

仍然无法避免反射的破解

枚举类

public enum EnumSingle {

    INSTANCE;
    private EnumSingle(){}
    public EnumSingle getInstance(){
        return INSTANCE;
    }
}

 

反射不能突破枚举类

在反射的Constructor 的newInstance方法中,限制了 通过反射创建枚举类。

枚举是真正的 final,客户端不允许创建枚举类的实例,也不能对其进行拓展

也就是枚举类不给外界实例化的机会,只能它自己实例化,而一个枚举类的所有实例就只有枚举前面分号前的那几个,其他地方不允许创建。

 

 

 

 

 

 

 

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值