模式回顧---單例

          最近突然想回顧一下設(shè)計模式,很多東西是要回過頭來總結(jié)一下的。今天先回顧一下單例吧。
          很多時候覺得挺搞笑的,去面試的時候如果人家問你設(shè)計模式,一般都是要你寫個單例模式。去年來北京好幾家面試都是問我這個。當(dāng)時我就想這個能反映出一個人的水平來嗎?還是說更多的是反映出這個公司的水平呢?
          隨著一年的應(yīng)用,很多地方都用過之后覺得,單例這個東西雖然簡單,可是現(xiàn)實(shí)是復(fù)雜的。所以單例這個簡單的模式也不能太小瞧咯。
          單例其實(shí)有很多種實(shí)現(xiàn),這是其中的一種,延遲加載的(好像英文叫Lazy?):
          [下面代碼中所有的構(gòu)造器都是私有的,這里我就省略不寫了。]
          public class ClassName {
              
          public static ClassName getInstance(){
                  
          if(instance == null)
                  {
                      instance 
          = new ClassName();
                  }
                  
          return instance;
              }
              
              
          private static ClassName instance;
          }
          這種的好處是我們的單例使用時才進(jìn)行初始化,這樣方便我們在系統(tǒng)啟動時做些小動作。但是這個方式不是線程安全的,想要完成一個線程安全的單例,有幾種方式:
          (一)
          public class ClassName {
              
          public static ClassName getInstance(){
                  
          return instance;
              }
              
              
          private static ClassName instance = new ClassName();
          }
          這種方式,可以保證我們的單例是線程安全的,畢竟我們唯一的實(shí)例在系統(tǒng)初始化的時候就構(gòu)造了。可是Java的機(jī)制是static級別的變量初始化時互相調(diào)用是會報異常的。所以隨著系統(tǒng)的擴(kuò)展,尤其還會有一些新手或者粗心大意的家伙(比如說,我)會亂用你的方法。一不小心就造成問題了。而且,你也失去了第一個方式中的一個小優(yōu)勢,不能在系統(tǒng)啟動時做點(diǎn)小動作了。
          (二)
          public class ClassName {
              
          public static synchronized ClassName getInstance(){
                  
          if(instance == null)
                  {
                       instance 
          = new ClassName();
                  }
                  
          return instance;
              }
              
              
          private static ClassName instance;
          }
          這樣倒是線程安全了,也可以延遲加載,但是從今以后這個getInstance方法就是synchronized的了,那絕對是很影響效率的。我跟朋友討論提出了幾種寫法,以期既能使單例可以在系統(tǒng)啟動不至于數(shù)據(jù)已經(jīng)煮成熟飯又是線程安全的:(少數(shù)人討論結(jié)果,代碼可能會比較丑陋,僅供參考,歡迎拍磚)

          public class ClassName {
              
          public static ClassName getInstance(){
                  
          if(instance == null)
                  {
                       instance 
          = ClassName.createInstance();
                  }
                  
          return instance;
              }
              
              
          private static synchronized ClassName createInstance(){
                  
          if(instance == null)
                  {    
                      
          return new ClassName();
                  }
          else{
                      
          return instance;
                  }
              }
              
              
          private static ClassName instance;
          }
          這種寫法就很好的解決了這些問題。
          還有一種寫法是這樣的,這個不是延遲加載的。而是采用了一種取巧的方式。
          public class ClassName {
              
          public static ClassName getInstance(){
                  
          if(!instance.isInit)
                  {
                       instance.initSingleton();
                  }
                  
          return instance;
              }
              
              
          private synchronized void initSingleton() {
                
          if(!isInit)
                {
                    reset();
          //這名字是有點(diǎn)怪異,我沒時間想太好聽的名字
                    isInit = true;
                }
              }
              
              
          public void  reset(){
                  
          //.....真正進(jìn)行數(shù)據(jù)初始化的地方
              }
              
              
          private boolean isInit = false;
              
              
          private static ClassName instance = new ClassName();
          }

          將所有的初始化代碼搬到構(gòu)造器之外。這是專為數(shù)據(jù)初始化和復(fù)位進(jìn)行的設(shè)計。所以我把reset開放了出來。



          posted on 2008-01-29 21:59 咖啡屋的鼠標(biāo) 閱讀(1451) 評論(7)  編輯  收藏 所屬分類: Java

          評論

          # re: 模式回顧---單例 2008-01-30 09:30 久城

          我一般用的都是這種:
          public class ClassName {
          public static ClassName getInstance(){
          if(instance == null)
          {
          instance = new ClassName();
          }
          return instance;
          }

          private static ClassName instance;
          }  回復(fù)  更多評論   

          # re: 模式回顧---單例 2008-01-30 10:05 大衛(wèi)

          不錯!  回復(fù)  更多評論   

          # re: 模式回顧---單例 2008-01-30 10:51 dennis

          public class ClassName {
          public static ClassName getInstance(){
          if(instance == null)
          {
          instance = ClassName.createInstance();
          }
          return instance;
          }

          private static synchronized ClassName createInstance(){
          return new ClassName();
          }

          private static ClassName instance;
          }
          這種寫法有問題的,可以想象下,線程A判斷instance==null,然后開始createInstance,正在此時,線程B也判斷instance==null,然后等待線程A完成創(chuàng)建并解鎖,然后線程B也createInstance,這還是單例嗎?單例模式的寫法網(wǎng)上已經(jīng)討論爛了,這樣的錯誤實(shí)在不應(yīng)該。  回復(fù)  更多評論   

          # re: 模式回顧---單例 2008-01-30 10:52 dennis

          第三種寫法就更無語了,靜態(tài)初始化,更談不上所謂的lazy initialize了。基礎(chǔ)還是好好看下吧。  回復(fù)  更多評論   

          # re: 模式回顧---單例 2008-01-30 10:54 dennis

          ps:我所說的第三種是最后那個。
          既然發(fā)表在blogjava首頁上,文章至少要保證沒有錯誤,不然誤導(dǎo)人也是害人害己,說的嚴(yán)重了,博主自己看看。  回復(fù)  更多評論   

          # re: 模式回顧---單例 2008-01-30 11:45 咖啡屋的鼠標(biāo)

          @dennis
          感謝您的批評指正,這個寫法是有點(diǎn)問題。昨天跟朋友討論出這個方法的時候沒考慮細(xì),剛才看了看,改改也能使用,可能就丑陋點(diǎn)了。先刪去。

          至于最后一種,確實(shí)不是lazy的,我看了看也沒寫是Lazy的,應(yīng)該不會有誤導(dǎo)概念的問題,但還是標(biāo)明了一下以防止誤會,很感謝你的指出。我只是把所有的初始化代碼放到了init里面,在構(gòu)造器里不做任何操作初始化操作,其實(shí)還應(yīng)該做一次是否初始化的二次判斷。省的多次初始化,但是那樣我就還要再寫一個函數(shù)來控制復(fù)位,說起來可能會更繁瑣,就沒加。

            回復(fù)  更多評論   

          # re: 模式回顧---單例 2008-01-30 13:18 咖啡屋的鼠標(biāo)

          考慮了一下,還是趁中午的時間把改正的結(jié)果更新了,經(jīng)過簡單的多線程測試是沒有問題的。  回復(fù)  更多評論   


          只有注冊用戶登錄后才能發(fā)表評論。


          網(wǎng)站導(dǎo)航:
           
          <2008年1月>
          303112345
          6789101112
          13141516171819
          20212223242526
          272829303112
          3456789

          導(dǎo)航

          統(tǒng)計

          常用鏈接

          留言簿(15)

          隨筆分類(52)

          隨筆檔案(76)

          文章分類(3)

          文章檔案(4)

          新聞檔案(1)

          收藏夾

          Flex

          搜索

          積分與排名

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 石渠县| 绩溪县| 沙洋县| 固始县| 汨罗市| 蒲江县| 金塔县| 平乐县| 富川| 永清县| 洛隆县| 辽源市| 临海市| 林甸县| 康乐县| 长寿区| 池州市| 竹北市| 威海市| 柳河县| 阳江市| 安平县| 张家界市| 通辽市| 株洲市| 孟村| 达孜县| 乐陵市| 林州市| 德阳市| 和政县| 堆龙德庆县| 横峰县| 慈利县| 罗平县| 金塔县| 隆安县| 孙吴县| 武宁县| 黑河市| 江城|