设计模式之单例模式(反射攻击)

模拟反射攻击

饿汉式模拟

饿汉式

public class HungrySingleton implements Serializable {
    private final static HungrySingleton hungrySingleton;
    static {
        hungrySingleton=new HungrySingleton();
    }
    private HungrySingleton(){
  if(HungrySingleton.getInstance()!=null){
      throw new RuntimeException("单例构造器禁止反射调用");
  }
    }
    public static HungrySingleton getInstance(){
        return hungrySingleton;
    }
    private Object readResolve(){
        return hungrySingleton;
    }
}

测试类

public class Test {
    public static void main(String[] args) throws Exception {

     Class objectClass=HungrySingleton.class;
        Constructor constructor=objectClass.getDeclaredConstructor();
        constructor.setAccessible(true);
        HungrySingleton newInstance=HungrySingleton.getInstance();
        HungrySingleton instance= (HungrySingleton) constructor.newInstance();
        System.out.println(instance);
        System.out.println(newInstance);
        System.out.println(instance==newInstance);
    }
}

结果如图:
在这里插入图片描述
原因在类加载时就完成初始化

懒汉式模拟

public class LazySingleton {
    //声明静态的要被单例的对象
    private static LazySingleton lazySingleton=null;

    private static boolean flag=true;

    //私有构造器,为了不让外部new
    private LazySingleton(){
        if(flag){
            flag=false;
        }else {
            throw new RuntimeException("单例构造器禁止反射调用");
        }

    }
    //加synchronized锁了整个类
    public synchronized static LazySingleton getInstance(){
        if(lazySingleton==null){
            lazySingleton=new LazySingleton();
        }
        return lazySingleton;
    }

    public static void main(String[] args) throws Exception {
        Class objClass=LazySingleton.class;
        Constructor constructor=objClass.getDeclaredConstructor();
        constructor.setAccessible(true);

        LazySingleton o1=LazySingleton.getInstance();
        //反射攻击,修改值
        Field flag=o1.getClass().getDeclaredField("flag");
        flag.setAccessible(true);
        flag.set(o1,true);

        LazySingleton o2= (LazySingleton) constructor.newInstance();

        System.out.println(o1);
        System.out.println(o2);
        System.out.println(o1==o2);
    }
}

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值