反射破坏单例 单例模式一般构造方法都是private,目的就是为了防止外界调用私有构造器创建多个实例,通过一个public的共有方法作为外界获取实例的唯一入口,从而实现单例。 但是反射能够访问私有的构造方法,只要反射获取的构造器调用setAccessible(true)方法即可。这样调用一次就会产生一个实例,调用多次就时多个实例,从而破坏单例。如何防止: 只要在单例的私有构造器中添加判断单例是否已经构造的代码,如果单例之前已经构造,则抛出异常,如果没有构造,则无所谓。如下单例被破坏则抛异常privateSingleTon1(){if(singleTon1!null){thrownewRuntimeException();}} 单例完整代码:publicclassSingleTon1{privatestaticSingleTon1singleTon1newSingleTon1();privateSingleTon1(){if(singleTon1!null){thrownewRuntimeException();}}publicstaticSingleTon1getSingleTon1(){returnsingleTon1;}}测试: 反射测试代码:publicstaticvoidmain(String〔〕args)throwsException{ConstructorSingleTon1constructorSingleTon1。class。getDeclaredConstructor();constructor。setAccessible(true);SingleTon1s1constructor。newInstance();SingleTon1s2constructor。newInstance();System。out。println(s1);System。out。println(s2);} 结果:抛出异常,阻止了调用私有构造器Exceptioninthreadmainjava。lang。reflect。InvocationTargetExceptionatsun。reflect。NativeConstructorAccessorImpl。newInstance0(NativeMethod)atsun。reflect。NativeConstructorAccessorImpl。newInstance(UnknownSource)atsun。reflect。DelegatingConstructorAccessorImpl。newInstance(UnknownSource)atjava。lang。reflect。Constructor。newInstance(UnknownSource)atsingleTonDestory。test。main(test。java:21)Causedby:java。lang。RuntimeExceptionatsingleTonDestory。SingleTon1。init(SingleTon1。java:13)。。。5more序列化破坏单例 添加readResolve方法就可以防止破坏单例publicclassSingleTon1implementsSerializable{privatestaticSingleTon1singleTon1newSingleTon1();privateSingleTon1(){if(singleTon1!null){thrownewRuntimeException();}}publicstaticSingleTon1getSingleTon1(){returnsingleTon1;}防止反序列化破坏单例,反序列化时,如果定义了readResolve方法,则直接返回吃方法指定的对象,而不需要单独再创建对象privateObjectreadResolve()throwsObjectStreamException{returnsingleTon1;}} 添加readResolve方法,反序列化时,会直接返回该方法指定的对象,不需要创建对象。测试: 序列化测试代码:publicstaticvoidmain(String〔〕args)throwsException{通过方法获取两个一样的单例SingleTon1s1SingleTon1。getSingleTon1();SingleTon1s2SingleTon1。getSingleTon1();序列化到磁盘(保存对象)FileOutputStreamfosnewFileOutputStream(d:a。txt);ObjectOutputStreamoosnewObjectOutputStream(fos);oos。writeObject(s1);oos。close();fos。close();从磁盘发序列化(获取对象)FileInputStreamfisnewFileInputStream(D:a。txt);ObjectInputStreamoisnewObjectInputStream(fis);SingleTon1s(SingleTon1)ois。readObject();System。out。println(s);System。out。println(s1);System。out。println(s2);} 结果:singleTonDestory。SingleTon15c647e05singleTonDestory。SingleTon15c647e05singleTonDestory。SingleTon15c647e05 从结果可以看出,防止了序列化破坏单例。 来源:blog。csdn。netweixin42130471articledetails89602999