詳解Java序列化如何破壞單例模式
先看代碼:
public class Singleton { private Singleton(){} private static class SingletonInstance{ private static final Singleton instance = new Singleton(); } public static Singleton getInstance(){ return SingletonInstance.instance; } }
我想應(yīng)該沒(méi)有不知道這行個(gè)類(lèi)是干嘛的小伙伴了吧,這是單例模式的一種寫(xiě)法。
單例模式是每一個(gè) Java boy 必須要掌握的設(shè)計(jì)模式,它所描述的是在某個(gè)進(jìn)程內(nèi),某個(gè)類(lèi)有且僅有一個(gè)實(shí)例。我們知道要破壞單例模式就必須讓它創(chuàng)建多個(gè)對(duì)象。創(chuàng)建對(duì)象的方式無(wú)非就幾種:
- new
- clone
- 反射
- 反序列化
首先單例模式的構(gòu)造器一定是 private 的,所以 new 這種方式是無(wú)法破壞單例模式的。 而 clone 需要實(shí)現(xiàn) Cloneable 接口,單例模式誰(shuí)如果實(shí)現(xiàn)了這個(gè)接口,請(qǐng)打死它。所以就剩下反射和反序列化了。本篇文章只討論反序列化。
反序列化破壞單例模式
與 clone 方式一樣,反序列化需要實(shí)現(xiàn) Serializable 接口,但是有小伙伴可能會(huì)說(shuō),誰(shuí)會(huì)在單例模式中實(shí)現(xiàn) Serializable 接口咯,除非他瘋了,確實(shí)是這種情況,但是在實(shí)際情況中它并不是一定會(huì)避免的,有些類(lèi)它就是一定要序列化。比如單例對(duì)象在不同環(huán)境或應(yīng)用實(shí)例之間的共享、持久化或狀態(tài)恢復(fù),當(dāng)然這些場(chǎng)景都屬于比較特殊的場(chǎng)景。
繼續(xù)用上面例子:
public class SerializableSingleton implements Serializable { // 省略部分代碼 }
然后在對(duì)該類(lèi)進(jìn)行序列化和反序列化
public class Test { public static void main(String[] args) throws Exception { ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("Singleton.txt")); SerializableSingleton singleton = SerializableSingleton.getInstance(); oos.writeObject(singleton); ObjectInputStream ois = new ObjectInputStream(new FileInputStream("Singleton.txt")); SerializableSingleton singleton1 = (SerializableSingleton) ois.readObject(); System.out.println("singleton = singleton1:" + (singleton == singleton1)); } }
運(yùn)行結(jié)果
singleton = singleton1:false
通過(guò)對(duì) Singleton 進(jìn)行反序列化得到了一個(gè)全新的對(duì)象,這就破壞了 Singleton 的單例性了。我們看 readObject()
源碼就知道了。
public final Object readObject() throws IOException, ClassNotFoundException { //... int outerHandle = passHandle; try { Object obj = readObject0(false); //... return obj; } finally { //... } }
調(diào)用 readObject0()
:
private Object readObject0(boolean unshared) throws IOException { // ... try { switch (tc) { // ... case TC_OBJECT: // readObject0() return checkResolve(readOrdinaryObject(unshared)); // ... } } finally { depth--; bin.setBlockDataMode(oldMode); } }
readObject0()
是根據(jù)反序列化對(duì)象的不同執(zhí)行不同的方法來(lái)反序列化一個(gè)實(shí)例對(duì)象。我們這里是 Object,所以進(jìn)一步看 readOrdinaryObject()
。
private Object readOrdinaryObject(boolean unshared) throws IOException { // ... Object obj; try { // 核心代碼 // 反射創(chuàng)建一個(gè)新對(duì)象 obj = desc.isInstantiable() ? desc.newInstance() : null; } catch (Exception ex) { throw (IOException) new InvalidClassException( desc.forClass().getName(), "unable to create instance").initCause(ex); } // ... if (obj != null && handles.lookupException(passHandle) == null && desc.hasReadResolveMethod()) { Object rep = desc.invokeReadResolve(obj); if (unshared && rep.getClass().isArray()) { rep = cloneArray(rep); } if (rep != obj) { // Filter the replacement object if (rep != null) { if (rep.getClass().isArray()) { filterCheck(rep.getClass(), Array.getLength(rep)); } else { filterCheck(rep.getClass(), -1); } } handles.setObject(passHandle, obj = rep); } } return obj; }
這段代碼最核心的地方就是:
obj = desc.isInstantiable() ? desc.newInstance() : null;
底層依然是利用反射的方式來(lái)創(chuàng)建一個(gè)新對(duì)象。
那么對(duì)于這種方式有什么保護(hù)措施沒(méi)?在 readOrdinaryObject()
最后面一段就已經(jīng)告知了:
if (obj != null && handles.lookupException(passHandle) == null && desc.hasReadResolveMethod()) { //.... }
判斷反序列化的類(lèi)是否已實(shí)現(xiàn)了 readResolve()
,如果有則會(huì)調(diào)用該方法,我們只需要在該方法里面返回原對(duì)象就可以了。驗(yàn)證下。
public class SerializableSingleton implements Serializable { // ... private Object readResolve() { return SingletonInstance.instance; } }
執(zhí)行結(jié)果:
singleton = singleton1:true
執(zhí)行結(jié)果為 true,就說(shuō)明序列化和反序列化出來(lái)的是同一個(gè)對(duì)象。
所以,要想防止單例被反序列化破壞,就讓單例實(shí)現(xiàn) readResolve()
方法,返回同一個(gè)對(duì)象即可。
到此這篇關(guān)于詳解Java序列化如何破壞單例模式的文章就介紹到這了,更多相關(guān)Java序列化內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java進(jìn)行反編譯生成.java文件方式(javap、jad下載安裝使用)
這篇文章主要介紹了Java進(jìn)行反編譯生成.java文件方式(javap、jad下載安裝使用),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12Java實(shí)現(xiàn)Map遍歷key-value的四種方法
本文主要介紹了Java實(shí)現(xiàn)Map遍歷key-value的四種方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07手把手教你寫(xiě)一個(gè)spring IOC容器的方法
這篇文章主要介紹了手把手教你寫(xiě)一個(gè)spring IOC容器的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04java8 stream多字段排序的實(shí)現(xiàn)
這篇文章主要介紹了java8 stream多字段排序的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03Nacos配置文件使用經(jīng)驗(yàn)及CAP原則詳解
這篇文章主要為大家介紹了Nacos配置文件使用經(jīng)驗(yàn)及CAP規(guī)則詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-02-02解決使用@ResponseBody后返回500錯(cuò)誤的問(wèn)題
這篇文章主要介紹了解決使用@ResponseBody后返回500錯(cuò)誤的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09JAVA根據(jù)ip地址獲取歸屬地的實(shí)現(xiàn)方法
本文主要介紹了JAVA根據(jù)ip地址獲取歸屬地的實(shí)現(xiàn)方法,要通過(guò)Java程序獲取IP地址對(duì)應(yīng)的城市,需要借助第三方的IP地址庫(kù),下面就來(lái)介紹一下,感興趣的可以了解一下2023-10-10spring cloud實(shí)現(xiàn)Eureka注冊(cè)中心的HA的方法
本篇文章主要介紹了spring cloud實(shí)現(xiàn)Eureka注冊(cè)中心的HA的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01