亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

JAVA破壞單例模式的方式以及避免方法

瀏覽:4日期:2022-08-31 08:51:36

單例模式,大家恐怕再熟悉不過了,其作用與實現方式有多種,這里就不??鋁?。?牽?勖竊謔褂謎廡┓絞絞迪值ダ?J絞保?絳蛑芯駝嫻幕嶂揮幸桓鍪道?穡?/p>

聰明的你看到這樣的問話,一定猜到了答案是NO。這里筆者就不賣關子了,開門見山吧!實際上,在有些場景下,如果程序處理不當,會無情地破壞掉單例模式,導致程序中出現多個實例對象。

下面筆者介紹筆者已知的三種破壞單例模式的方式以及避免方法。

1、反射對單例模式的破壞

我們先通過一個例子,來直觀感受一下

(1)案例

DCL實現的單例模式:

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

測試代碼:

public class SingletonDemo { public static void main(String[] args){ Singleton singleton = Singleton.getInstance(); try { Constructor<Singleton> constructor = Singleton.class.getDeclaredConstructor(); constructor.setAccessible(true); Singleton reflectSingleton = constructor.newInstance(); System.out.println(reflectSingleton == singleton); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } }}

執行結果:

false

運行結果說明,采用反射的方式另辟蹊徑實例了該類,導致程序中會存在不止一個實例。

(2)解決方案

其思想就是采用一個全局變量,來標記是否已經實例化過了,如果已經實例化過了,第二次實例化的時候,拋出異常。實現代碼如下:

public class Singleton{ private static volatile Singleton mInstance; private static volatile boolean mIsInstantiated = false; private Singleton(){ if (mIsInstantiated){ throw new RuntimeException('Has been instantiated, can not do it again!'); } mIsInstantiated = true; } public static Singleton getInstance(){ if(mInstance == null){ synchronized (Singleton.class) {if(mInstance == null){ mInstance = new Singleton();} } } return mInstance; }}

執行結果:

JAVA破壞單例模式的方式以及避免方法

2、clone()對單例模式的破壞

當需要實現單例的類允許clone()時,如果處理不當,也會導致程序中出現不止一個實例。

(1)案例

一個實現了Cloneable接口單例類:

public class Singleton implements Cloneable{ private static volatile Singleton mInstance; private Singleton(){ } public static Singleton getInstance(){ if(mInstance == null){ synchronized (Singleton.class) {if(mInstance == null){ mInstance = new Singleton();} } } return mInstance; } @Override protected Object clone() throws CloneNotSupportedException { // TODO Auto-generated method stub return super.clone(); }}

測試代碼:

public class SingletonDemo { public static void main(String[] args){ try { Singleton singleton = Singleton.getInstance(); Singleton cloneSingleton; cloneSingleton = (Singleton) Singleton.getInstance().clone(); System.out.println(cloneSingleton == singleton); } catch (CloneNotSupportedException e) { e.printStackTrace(); } }}

執行結果:

false

(2)解決方案:

解決思想是,重寫clone()方法,調clone()時直接返回已經實例的對象

public class Singleton implements Cloneable{ private static volatile Singleton mInstance; private Singleton(){ } public static Singleton getInstance(){ if(mInstance == null){ synchronized (Singleton.class) {if(mInstance == null){ mInstance = new Singleton();} } } return mInstance; } @Override protected Object clone() throws CloneNotSupportedException { return mInstance; }}

執行結果:

true

3、序列化對單例模式的破壞

在使用序列化/反序列化時,也會出現產生新實例對象的情況。

(1)案例

一個實現了序列化接口的單例類:

public class Singleton implements Serializable{ private static volatile Singleton mInstance; private Singleton(){ } public static Singleton getInstance(){ if(mInstance == null){ synchronized (Singleton.class) {if(mInstance == null){ mInstance = new Singleton();} } } return mInstance; }}

測試代碼:

public class SingletonDemo { public static void main(String[] args){ try { Singleton singleton = Singleton.getInstance(); FileOutputStream fos = new FileOutputStream('singleton.txt'); ObjectOutputStream oos = new ObjectOutputStream(fos); oos.writeObject(singleton); oos.close(); fos.close(); FileInputStream fis = new FileInputStream('singleton.txt'); ObjectInputStream ois = new ObjectInputStream(fis); Singleton serializedSingleton = (Singleton) ois.readObject(); fis.close(); ois.close(); System.out.println(serializedSingleton==singleton); } catch (Exception e) { e.printStackTrace(); } }}

運行結果:

false

(2)解決方案

在反序列化時的回調方法 readResolve()中返回單例對象。

public class Singleton implements Serializable{ private static volatile Singleton mInstance; private Singleton(){ } public static Singleton getInstance(){ if(mInstance == null){ synchronized (Singleton.class) {if(mInstance == null){ mInstance = new Singleton();} } } return mInstance; } protected Object readResolve() throws ObjectStreamException{ return mInstance; }}

結果:

true

以上就是筆者目前已知的三種可以破壞單例模式的場景以及對應的解決辦法,讀者如果知道還有其他的場景,記得一定要分享出來噢,正所謂“獨樂樂不如眾樂樂”?。?!

單例模式看起來是設計模式中最簡單的一個,但“麻雀雖小,五臟俱全”,其中有很多細節都是值得深究的。即便是本篇介紹的這幾個場景,也只是介紹了一些梗概而已,很多細節還需要讀者自己去試驗和推敲的,比如:通過枚舉方式實現單例模式,就不存在上述問題,而其它的實現方式似乎都存在上述問題!

后記

本篇參(剽)考(竊)了如下資料:https://www.jb51.net/article/143047.htm

以上就是JAVA破壞單例模式的方式以及避免方法的詳細內容,更多關于JAVA 單例模式的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
主站蜘蛛池模板: 欧美视频在线观看一区二区 | 亚洲视频一区在线观看 | 黄色一级免费片 | 在线播放国产精品 | 久久亚洲精品中文字幕亚瑟 | 久久久一区二区三区不卡 | 91小视频版在线观看www | 色黄啪啪18周岁以下禁止观看 | 国产一级做a爰片久久毛片99 | 日本高清动作片www欧美 | 可以在线看黄的网站 | 狠狠色丁香婷婷综合激情 | 日韩做爰在线观看免费 | 三级黄.色 | 国产精品免费观看网站 | 国产91福利在线精品剧情尤物 | 国产在线视频www色 国产在线视频一区 | 97视频免费公开成人福利 | 日韩第一视频 | 麻豆igao在线视频 | 国产在线色视频 | 日韩 欧美 亚洲国产 | 国产精品v欧美精品v日韩精品 | 国产色婷婷视频在线观看 | 亚洲精品网站在线 | 亚洲三级成人 | 小明台湾成人永久免费看看 | 免费一级毛片在线播放傲雪网 | 日韩在线小视频 | 中国日韩欧美中文日韩欧美色 | 日本欧洲95视频 | 国产成人黄网在线免 | 成人深爱网 | 日本aaa级片| 国产伦精品一区二区三区女 | 91成人午夜性a一级毛片 | 欧美日韩一区二区亚洲 | 日本久久综合视频 | 国产乱理伦片a级在线观看 国产乱理伦片在线观看 | 成人免费福利片在线观看 | 国产精品大白天新婚身材 |