单例模式破坏和解决方案

单例模式破坏和解决方案

1.破坏

  1. 序列化
  2. 克隆
  3. 反射

2.破坏实战

package com.jun.design.pattern;

import java.io.*;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Singleton implements Serializable,Cloneable {

    private static final long serialVersionUID = -3465901519138664777L;

    private Singleton() {}

    private static volatile  Singleton instance;

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

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }

    public static void main(String[] args) throws IOException, ClassNotFoundException, CloneNotSupportedException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        System.out.println("序列化:");
        Singleton originSingleton = Singleton.getInstance();
        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);
        objectOutputStream.writeObject(Singleton.getInstance());
        ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
        ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
        Singleton serializeSingleton = (Singleton) objectInputStream.readObject();
        //false
        System.out.println(originSingleton == serializeSingleton);

        System.out.println("克隆:");
        Singleton cloneSingleton = (Singleton) originSingleton.clone();
        //false
        System.out.println(cloneSingleton == originSingleton);

        //通过反射获取
        System.out.println("反射:");
        Constructor<Singleton> declaredConstructor = Singleton.class.getDeclaredConstructor();
        declaredConstructor.setAccessible(true);
        Singleton reflectSingleton = declaredConstructor.newInstance();
        //false
        System.out.println(originSingleton == reflectSingleton);
    }
}

3.解决方案

  1. 防止序列化破坏 ,添加readResolve(),返回Object对象
  2. 防止克隆破坏,重写clone(),直接返回单例对象
  3. 防止反射,定义一个全局变量,当第二次创建的时候抛出异常
 package com.jun.design.pattern;

import java.io.*;
import java.lang.reflect.Constructor;

public class SingletonResolve implements Serializable,Cloneable { 
   private static final long serialVersionUID = -2144773240242119659L;
    //默认是第一次创建
   private static volatile boolean isCreate = false;

   private SingletonResolve(){
       if(isCreate){
           throw new RuntimeException("已经被实例化一次,不能在实例化");
       }
       isCreate = true;
   }
   private static  volatile  SingletonResolve instance;

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

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return instance;
    }

    //防止序列化破环
    private Object readResolve() {
        return instance;
    }

    public static void main(String[] args) throws Exception{
        System.out.println("序列化:");
        SingletonResolve originSingletonResolve = SingletonResolve.getInstance();
        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);
        objectOutputStream.writeObject(SingletonResolve.getInstance());
        ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
        ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
        SingletonResolve serializeSingletonResolve = (SingletonResolve) objectInputStream.readObject();
        //true
        System.out.println(originSingletonResolve == serializeSingletonResolve);

        System.out.println("克隆:");
        SingletonResolve cloneSingletonResolve = (SingletonResolve) originSingletonResolve.clone();
        //true
        System.out.println(originSingletonResolve == cloneSingletonResolve);

        System.out.println("反射:");
        Constructor<SingletonResolve> declaredConstructor = SingletonResolve.class.getDeclaredConstructor();
        declaredConstructor.setAccessible(true);
        //Caused by: java.lang.RuntimeException: 已经被实例化一次,不能在实例化
        SingletonResolve reflectDeclaredConstructor = declaredConstructor.newInstance();
        System.out.println(originSingletonResolve == reflectDeclaredConstructor);
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值