防止单例模式被反射和单例模式破坏

反射破坏单例:

单例模式一般构造方法都是private,目的就是为了防止外界调用私有构造器创建多个实例,通过一个public的共有方法作为外界获取实例的唯一入口。从而实现单例。

但是反射能够访问私有的构造方法,只要反射获取的构造器调用setAccessible(true)方法即可。这样调用一次就会产生一个实例,调用多次就时多个实例,从而破坏单例。

如何防止:

只要在单例的私有构造器中添加判断单例是否已经构造的代码,如果单例之前已经构造,则抛出异常,如果没有构造,则无所谓。如下

    //单例被破坏则抛异常
	private SingleTon1() {
		if(singleTon1 != null) {
			throw new RuntimeException();
		}
	}

 单例完整代码:

public class SingleTon1 {
	private static SingleTon1 singleTon1 = new SingleTon1();
	
	private SingleTon1() {
		if(singleTon1 != null) {
			throw new RuntimeException();
		}
	}

	public static SingleTon1 getSingleTon1() {
		return singleTon1;
	}
}

测试:

1.反射测试代码: 

public static void main(String[] args) throws Exception {
                Constructor<SingleTon1> constructor=SingleTon1.class.getDeclaredConstructor();
		constructor.setAccessible(true);
		SingleTon1 s1=constructor.newInstance();
		SingleTon1 s2=constructor.newInstance();
		System.out.println(s1);
		System.out.println(s2);
}

结果:抛出异常,阻止了调用私有构造器 

Exception in thread "main" java.lang.reflect.InvocationTargetException
	at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
	at sun.reflect.NativeConstructorAccessorImpl.newInstance(Unknown Source)
	at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(Unknown Source)
	at java.lang.reflect.Constructor.newInstance(Unknown Source)
	at singleTonDestory.test.main(test.java:21)
Caused by: java.lang.RuntimeException
	at singleTonDestory.SingleTon1.<init>(SingleTon1.java:13)
	... 5 more

序列化破坏单例:

添加readResolve方法就可以防止破坏单例

public class SingleTon1 implements Serializable{
	private static SingleTon1 singleTon1 = new SingleTon1();
	
	private SingleTon1() {
		if(singleTon1 != null) {
			throw new RuntimeException();
		}
	}
	public static SingleTon1 getSingleTon1() {
		return singleTon1;
	}
	
	//防止反序列化破坏单例,反序列化时,如果定义了readResolve方法,则直接返回吃方法指定的对象,而不需要单独再创建对象
	private Object readResolve() throws ObjectStreamException{
		return singleTon1;
	}
}

添加readResolve方法,反序列化时,会直接返回该方法指定的对象,不需要创建对象。

 

2.序列化测试代码 

public static void main(String[] args) throws Exception {
 //通过方法获取两个一样的单例
		SingleTon1 s1 = SingleTon1.getSingleTon1();
		SingleTon1 s2 = SingleTon1.getSingleTon1();
		
		//序列化到磁盘(保存对象)
		FileOutputStream fos = new FileOutputStream("d:/a.txt");
		ObjectOutputStream oos = new ObjectOutputStream(fos);
		oos.writeObject(s1);
		oos.close();
		fos.close();

		//从磁盘发序列化(获取对象)
		FileInputStream fis = new FileInputStream("D:/a.txt");
		ObjectInputStream ois = new ObjectInputStream(fis);
		SingleTon1 s= (SingleTon1) ois.readObject();

		System.out.println(s);
		System.out.println(s1);
		System.out.println(s2);
}

结果:

singleTonDestory.SingleTon1@5c647e05
singleTonDestory.SingleTon1@5c647e05
singleTonDestory.SingleTon1@5c647e05

从结果可以看出,防止了序列化破坏单例。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值