亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

菜鳥學習java設計模式之單例模式

2024-07-13 10:13:43
字體:
來源:轉載
供稿:網友

單例模式大家并不陌生,也都知道它分為什么懶漢式、餓漢式之類的。但是你對單例模式的理解足夠透徹嗎?今天我帶大家一起來看看我眼中的單例,可能會跟你的認識有所不同。

下面是一個簡單的小實例:

//簡單懶漢式 public class Singleton {      //單例實例變量   private static Singleton instance = null;      //私有化的構造方法,保證外部的類不能通過構造器來實例化   private Singleton() {}      //獲取單例對象實例   public static Singleton getInstance() {          if (instance == null) {        instance = new Singleton();      }          System.out.println("我是簡單懶漢式單例!");     return instance;   } } 

很容易看出,上面這段代碼在多線程的情況下是不安全的,當兩個線程進入if (instance == null)時,兩個線程都判斷instance為空,接下來就會得到兩個實例了。這不是我們想要的單例。

接下來我們用加鎖的方式來實現互斥,從而保證單例的實現。

//同步法懶漢式 public class Singleton {      //單例實例變量   private static Singleton instance = null;      //私有化的構造方法,保證外部的類不能通過構造器來實例化   private Singleton() {}      //獲取單例對象實例   public static synchronized Singleton getInstance() {          if (instance == null) {        instance = new Singleton();      }          System.out.println("我是同步法懶漢式單例!");     return instance;   } } 

加上synchronized后確實保證了線程安全,但是這樣就是最好的方法嗎?很顯然它不是,因為這樣一來每次調用getInstance()方法是都會被加鎖,而我們只需要在第一次調用getInstance()的時候加鎖就可以了。這顯然影響了我們程序的性能。我們繼續尋找更好的方法。

經過分析發現,只需要保證instance = new Singleton()是線程互斥就可以保證線程安全,所以就有了下面這個版本:

//雙重鎖定懶漢式 public class Singleton {      //單例實例變量   private static Singleton instance = null;      //私有化的構造方法,保證外部的類不能通過構造器來實例化   private Singleton() {}      //獲取單例對象實例   public static Singleton getInstance() {     if (instance == null) {        synchronized (Singleton.class) {         if (instance == null) {            instance = new Singleton();          }       }     }     System.out.println("我是雙重鎖定懶漢式單例!");     return instance;   } } 

這次看起來既解決了線程安全問題,又不至于每次調用getInstance()都會加鎖導致降低性能。看起來是一個完美的解決方案,事實上是這樣的嗎?

很遺憾,事實并非我們想的那么完美。java平臺內存模型中有一個叫“無序寫”(out-of-order writes)的機制。正是這個機制導致了雙重檢查加鎖方法的失效。這個問題的關鍵在上面代碼上的第5行:instance = new Singleton(); 這行其實做了兩個事情:1、調用構造方法,創建了一個實例。2、把這個實例賦值給instance這個實例變量。可問題就是,這兩步jvm是不保證順序的。也就是說。可能在調用構造方法之前,instance已經被設置為非空了。下面我們一起來分析一下:

假設有兩個線程A、B

1、線程A進入getInstance()方法。
2、因為此時instance為空,所以線程A進入synchronized塊。
3、線程A執行 instance = new Singleton(); 把實例變量instance設置成了非空。(注意,是在調用構造方法之前。)
4、線程A退出,線程B進入。
5、線程B檢查instance是否為空,此時不為空(第三步的時候被線程A設置成了非空)。線程B返回instance的引用。(問題出現了,這時instance的引用并不是Singleton的實例,因為沒有調用構造方法。)
6、線程B退出,線程A進入。
7、線程A繼續調用構造方法,完成instance的初始化,再返回。

難道就沒有一個好方法了嗎?好的方法肯定是有的,我們繼續探索!

//解決無序寫問題懶漢式 public class Singleton {      //單例實例變量   private static Singleton instance = null;      //私有化的構造方法,保證外部的類不能通過構造器來實例化   private Singleton() {}      //獲取單例對象實例   public static Singleton getInstance() {     if (instance == null) {        synchronized (Singleton.class) {         //1         Singleton temp = instance;        //2         if (temp == null) {           synchronized (Singleton.class) { //3              temp = new Singleton();  //4             }           instance = temp;         //5            }       }     }     System.out.println("我是解決無序寫懶漢式單例!");     return instance;   }   } 

1、線程A進入getInstance()方法。
2、因為instance是空的 ,所以線程A進入位置//1的第一個synchronized塊。
3、線程A執行位置//2的代碼,把instance賦值給本地變量temp。instance為空,所以temp也為空。
4、因為temp為空,所以線程A進入位置//3的第二個synchronized塊。(后來想想這個鎖有點多余)
5、線程A執行位置//4的代碼,把temp設置成非空,但還沒有調用構造方法!(“無序寫”問題)
6、如果線程A阻塞,線程B進入getInstance()方法。
7、因為instance為空,所以線程B試圖進入第一個synchronized塊。但由于線程A已經在里面了。所以無法進入。線程B阻塞。
8、線程A激活,繼續執行位置//4的代碼。調用構造方法。生成實例。
9、將temp的實例引用賦值給instance。退出兩個synchronized塊。返回實例。
10、線程B激活,進入第一個synchronized塊。
11、線程B執行位置//2的代碼,把instance實例賦值給temp本地變量。
12、線程B判斷本地變量temp不為空,所以跳過if塊。返回instance實例。

到此為止,上面的問題我們是解決了,但是我們突然發現為了解決線程安全問題,但給人的感覺就像身上纏了很多毛線.... 亂糟糟的,所以我們要精簡一下:

//餓漢式 public class Singleton {      //單例變量 ,static的,在類加載時進行初始化一次,保證線程安全    private static Singleton instance = new Singleton();        //私有化的構造方法,保證外部的類不能通過構造器來實例化。      private Singleton() {}      //獲取單例對象實例      public static Singleton getInstance() {     System.out.println("我是餓漢式單例!");     return instance;   } } 

看到上面的代碼,瞬間覺得這個世界清靜了。不過這種方式采用的是餓漢式的方法,就是預先聲明Singleton對象,這樣帶來的一個缺點就是:如果構造的單例很大,構造完又遲遲不使用,會導致資源浪費。

到底有沒有完美的方法呢?繼續看:

//內部類實現懶漢式 public class Singleton {      private static class SingletonHolder{     //單例變量      private static Singleton instance = new Singleton();   }      //私有化的構造方法,保證外部的類不能通過構造器來實例化。   private Singleton() {        }      //獲取單例對象實例   public static Singleton getInstance() {     System.out.println("我是內部類單例!");     return SingletonHolder.instance;   } } 

懶漢式(避免上面的資源浪費)、線程安全、代碼簡單。因為java機制規定,內部類SingletonHolder只有在getInstance()方法第一次調用的時候才會被加載(實現了lazy),而且其加載過程是線程安全的(實現線程安全)。內部類加載的時候實例化一次instance。

簡單說一下上面提到的無序寫,這是jvm的特性,比如聲明兩個變量,String a; String b; jvm可能先加載a也可能先加載b。同理,instance = new Singleton();可能在調用Singleton的構造函數之前就把instance置成了非空。這是很多人會有疑問,說還沒有實例化出Singleton的一個對象,那么instance怎么就變成非空了呢?它的值現在是什么呢?想了解這個問題就要明白instance = new Singleton();這句話是怎么執行的,下面用一段偽代碼向大家解釋一下:

mem = allocate();       //為Singleton對象分配內存。 instance = mem;        //注意現在instance是非空的,但是還沒有被初始化。  ctorSingleton(instance);  //調用Singleton的構造函數,傳遞instance. 

由此可見當一個線程執行到instance = mem; 時instance已為非空,如果此時另一個線程進入程序判斷instance為非空,那么直接就跳轉到return instance;而此時Singleton的構造方法還未調用instance,現在的值為allocate();返回的內存對象。所以第二個線程得到的不是Singleton的一個對象,而是一個內存對象。

以上就是就是我對單例模式的一點小小的思考跟理解,熱烈歡迎各位大神前來指導批評。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产一区二中文字幕在线看| 国产精品入口免费视频一| 色综合91久久精品中文字幕| 久久久久国产精品www| 日韩av一区二区在线| 国产精品一区二区三| 日韩中文第一页| 精品国内亚洲在观看18黄| 欧美大片大片在线播放| 欧美日韩国产在线看| 欧美在线免费看| 欧美性生交大片免费| 色视频www在线播放国产成人| 欧美大片在线看免费观看| 精品久久久91| 日韩国产激情在线| 精品久久久中文| 国产精品精品久久久| 亚洲精品在线观看www| 欧美日韩不卡合集视频| 中文字幕欧美精品在线| 三级精品视频久久久久| 91精品国产乱码久久久久久蜜臀| 国产精品入口尤物| 亚洲久久久久久久久久久| 亚洲成色999久久网站| 国产精品自产拍在线观看| 成人写真福利网| 日韩欧美亚洲综合| 成人免费xxxxx在线观看| 日韩免费观看高清| 亚洲韩国青草视频| 亚洲性无码av在线| 97视频在线观看播放| 日韩欧美精品在线观看| 欧美日韩激情小视频| 欧洲精品毛片网站| 97国产成人精品视频| 久久精品国产欧美亚洲人人爽| www日韩欧美| 久久精品久久久久| 久久视频这里只有精品| 亚洲欧美一区二区三区情侣bbw| 欧美日韩在线第一页| 精品美女久久久久久免费| 亚洲深夜福利网站| 成人免费看黄网站| 激情成人中文字幕| 国产精品日韩欧美综合| 欧美精品成人91久久久久久久| 在线日韩第一页| 久久夜色精品国产亚洲aⅴ| 在线日韩第一页| 中文字幕亚洲一区二区三区五十路| xxxxx91麻豆| 疯狂做受xxxx高潮欧美日本| 中文字幕亚洲欧美日韩2019| 国产深夜精品福利| 色妞欧美日韩在线| 欧美成人免费小视频| 久久精品国产亚洲精品2020| 91麻豆国产精品| 亚洲精品视频中文字幕| 日韩中文有码在线视频| 国产午夜精品久久久| 久久91精品国产91久久跳| 久久偷看各类女兵18女厕嘘嘘| 国内精品国产三级国产在线专| 色综合老司机第九色激情| 亚洲免费人成在线视频观看| 免费av在线一区| 欧美一级大片视频| 日韩精品久久久久久久玫瑰园| 日韩视频在线免费| 亚洲综合中文字幕在线观看| 亚洲va欧美va在线观看| 国产亚洲精品美女久久久久| 91久久久久久久久久| 国产精品中文字幕久久久| 成人av.网址在线网站| 亚洲欧洲国产精品| 亚洲精选一区二区| 蜜臀久久99精品久久久久久宅男| 国产精品久久97| 欧美黑人性视频| 亚洲精品成人免费| 亚洲免费伊人电影在线观看av| 在线播放亚洲激情| 久久国产精品影片| 久99久在线视频| 亚洲国产欧美一区二区三区同亚洲| 国产精品v片在线观看不卡| 奇米四色中文综合久久| 国产成人鲁鲁免费视频a| 91久久综合亚洲鲁鲁五月天| 亚洲成人激情在线| 欧美激情手机在线视频| 国产亚洲欧美另类中文| 欧美黄网免费在线观看| 欧美性20hd另类| 精品国产91久久久久久| 欧美午夜性色大片在线观看| 96精品久久久久中文字幕| 久久精品国产亚洲精品2020| 国产精品国产三级国产aⅴ9色| 久久中文字幕在线视频| 久久深夜福利免费观看| 成人乱人伦精品视频在线观看| 欧美精品手机在线| 欧美激情综合亚洲一二区| 日韩成人激情视频| 欧美又大粗又爽又黄大片视频| 97香蕉超级碰碰久久免费软件| 亚洲国产成人爱av在线播放| 懂色av一区二区三区| 亚洲人成在线一二| 一区二区三区视频免费| 欧美视频在线免费看| 欧美日韩中文在线观看| 欧美日韩国产成人| 欧美精品久久久久久久| 欧美成在线视频| 亚洲欧洲在线免费| 久久久久久成人| 欧美日韩国产精品一区二区三区四区| 亚洲自拍偷拍一区| 一区二区三区国产在线观看| 欧美电影免费观看高清| 日韩动漫免费观看电视剧高清| 亚洲视频精品在线| 欧美电影在线观看高清| 亚洲欧美国产精品va在线观看| 日韩一级黄色av| 日韩av中文字幕在线免费观看| 国产精品电影久久久久电影网| 成人欧美一区二区三区黑人| 国产精品久久色| 欧美孕妇毛茸茸xxxx| 日韩在线视频网站| 亚洲欧美国产va在线影院| 国产精品视频区1| 亚洲美女在线视频| 伊人久久免费视频| 456国产精品| 97视频在线观看免费高清完整版在线观看| 日韩精品免费在线| 日本三级韩国三级久久| 亚洲在线视频福利| 国产成人精品久久二区二区| 国产精品极品美女粉嫩高清在线| 成人在线播放av| 国产欧美在线视频| 亚洲欧美国产精品| 韩国精品久久久999| 91精品在线观| 日本亚洲欧洲色α| 一区二区三区视频免费| 亚洲欧美国产日韩中文字幕| 国产一区二区三区在线| 一区二区欧美亚洲| 91精品久久久久久久久中文字幕| 亚洲精品福利在线| 亚洲香蕉av在线一区二区三区| 国产精品扒开腿做爽爽爽的视频|