<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    回歸

    驀然回首,那人卻在燈火闌珊處

      BlogJava :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
      4 Posts :: 2 Stories :: 12 Comments :: 0 Trackbacks
          現(xiàn)在大多數(shù)Java軟件工程師面試都會問到這個問題:什么是單例模式(Singleton),能否寫出單例模式的示例代碼?單例模式是Gof中23個模式中最簡單最容易入門的模式,學習它我們能更理性的感知模式的意義.

         [形成]

          Singleton Pattern 為什么會出現(xiàn)?在我們軟件開發(fā)和架構(gòu)中,經(jīng)常遇到這樣的情形:我們需要一個類只能且僅能產(chǎn)生一個實例..比如表示一臺計算機的類,表示系統(tǒng)設(shè)定的類或者是表示窗口的類,還有為了節(jié)約資源,只讓產(chǎn)生一個實例..
         
          如何構(gòu)造這種情形?Singleton模式給我一個方案:


         [代碼示例]
          
          程序列表
       

    名稱

    說明

    Singleton

    只有一個對象實例的類

    Main

    測試用的類



          [UML圖]
       
          

    [示例代碼和類的詮釋]

     1 package singleton;
     2 
     3 public class Singleton {
     4     private static Singleton singleton = new Singleton();
     5 
     6     private Singleton() {
     7         System.out.println("Create instance...");
     8     }
     9 
    10     public static Singleton getInstance() {
    11         return singleton;
    12     }
    13 }
    14 


              Singleton  Class:
                
                1.該類只能產(chǎn)生一個對象實例

                2.把該類的的singleton屬性設(shè)定為static再以Singleton;類的對象實例進行初始化,這個初始化的過程僅加載Sington類的時候調(diào)用一次.(Line4) 

                3.把Singleton 類的構(gòu)造函數(shù)限定為private,目的是為了防止從非Singleton類(其他類)調(diào)用構(gòu)造函數(shù)來產(chǎn)生實例,如果通過new方式來產(chǎn)生Singleton實例,會出現(xiàn)編譯錯誤.這樣做是為了保險己見.(Line6) 

                4.要得到Singleton實例的唯一方法就是調(diào)用類靜態(tài)方法getInstance().這個名字可以隨便取,只要方便理解就行.(Line 10) 

     1 package singleton;
     2 
     3 public class Main {
     4 
     5     public static void main(String[] args) {
     6         System.out.println("Start");
     7         Singleton obj1 = Singleton.getInstance();
     8         Singleton obj2 = Singleton.getInstance();
     9         if(obj1 == obj2){
    10             System.out.println("obj1和obj2是同一個對象實例");
    11         }else{
    12             System.out.println("obj1和obj2不是同一個對象實例");
    13         }
    14         System.out.println("End");
    15     }
    16 }
    17 


           Main Class
            1.該類是測試程序.
            2.程序通過getInstance()方式產(chǎn)生兩個obj1和obj2實例.(Line 7,Line 8)
            3.通過ojb1= =ojb2表達式來確定兩個對象是否相同,判定是否產(chǎn)生了Singleton的第二個示例.(Line9-12)
     

    示例程序的執(zhí)行結(jié)果
    Start
    Create instance...
    obj1和obj2是同一個對象實例
    End

             執(zhí)行結(jié)果含義:
                    1. 的確如此,obj1和obj2是Singleton類的同一個且唯一的對象實例. 
                    2.當程序執(zhí)行后,第一次調(diào)用getInstance的時候會初始化Singleton類,同時也會初始化static字段,也同時產(chǎn)生產(chǎn)生了一個唯一對象實例.




          [拓展思考]
       
    如下的另一一個單例模式的程序有什么隱患?

     1 package singleton;
     2 
     3 public class Singleton2 {
     4     
     5     private static Singleton2 singleton = null;
     6     
     7     private Singleton2(){
     8         System.out.println("已產(chǎn)生對象實例");
     9     }
    10     public static Singleton2 getInstance(){
    11         if(singleton == null){
    12             singleton = new Singleton2();
    13         }
    14         return singleton;
    15     }
    16 
    17 }
    18 

        [解答]
    當多線程同時調(diào)用Singleton2.getInstance()方法時,可能會產(chǎn)生多個對象實例,例如
    public class Main extends Thread{

        
    public static void main(String[] args) {
            System.out.println(
    "Starts.");
            
    new Main("A").start();
            
    new Main("B").start();
            
    new Main("C").start();
            System.out.println(
    "End.");
        }

       
        
    public void run(){
            Singleton2 obj 
    = Singleton2.getInstance();
            System.out.println(getName()
    +": obj ="+obj);
        }

        
        
    public Main(String name){
            
    super(name);
        }

    }




    public class Singleton2 {
        
    private static Singleton2 singleton2 = null;
        
    private Singleton2(){
            System.out.println(
    "已產(chǎn)生對象實例");
            solwDown();
        }


        
    public static Singleton2 getInstance(){
            
    if(singleton2 == null){
                singleton2 
    = new Singleton2();
            }

            
    return singleton2;
        }

        
         
    private  void solwDown(){
             
    try{
                 Thread.sleep(
    1000);
             }
    catch(InterruptedException e){
                 e.printStackTrace();
             }

         }

    }


    執(zhí)行結(jié)果:

    Start.
    End.
    已產(chǎn)生對象實例.
    已產(chǎn)生對象實例.
    已產(chǎn)生對象實例.
    B: obj = Singleton2#2a9348
    C: obj = Singleton2#b91134
    A: obj = Singleton2#e343l12

    (#替換為@)

    之所以會知道這種情況是因為if(singleton = = null){ singleton = new Singleton2(); }判斷不夠嚴謹?shù)膶е隆?br /> 利用: singleton == null 判斷為空后去執(zhí)行new Singleton2()之前,可能會有其他線程來搶先判斷表達式singleton == null,從而又執(zhí)行一遍創(chuàng)建實例的操作。

    解決辦法:
    給getInstance()方法添加Synchronized修飾符,即可修改成線程安全嚴謹?shù)膯卫J健?br />
    public class Singleton2 {
        
    private static Singleton2 singleton = null;

        
    private Singleton2() {
            System.out.println(
    "已產(chǎn)生對象實例");
            solwDown();
        }

        
    public static synchronized Singleton2 getInstance() {
            
    if (singleton == null) {
                singleton 
    = new Singleton2();
            }
            
    return singleton;
        }

        
    private void solwDown() {
            
    try {
                Thread.sleep(
    1000);
            } 
    catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    posted on 2007-06-08 11:36 回歸 閱讀(1216) 評論(7)  編輯  收藏 所屬分類: 個人原創(chuàng)

    Feedback

    # re: 親密接觸設(shè)計模式(二)-------Singleton(單例) 2007-06-08 12:57 dennis
    第一個示例少了static
    public static Singleton getInstance() {
    return singleton;
    }
    也沒有考慮多線程情況下的單例模式  回復  更多評論
      

    # re: 親密接觸設(shè)計模式(二)-------Singleton(單例) 2007-06-08 16:50 sitinspring
    多線程情況下的單例模式:

    1 public class Singleton{
    2 private static Singleton instance=null;
    3
    4 public static synchronized Singleton getInstance(){
    5 // 要用的時候再把Singleton建立起來
    6 if(instance==null){
    7 instance=new Singleton();
    8 }
    9
    10 return instance;
    11 }
    12 }


      回復  更多評論
      

    # re: 親密接觸設(shè)計模式(二)-------Singleton(單例) 2007-06-11 09:17 一點思想
    第一種實行的Singleton不存在多線程隱患,累以第一次加載時候就產(chǎn)生實例.你認為呢?  回復  更多評論
      

    # re: 親密接觸設(shè)計模式(二)-------Singleton(單例) 2007-06-11 09:22 一點思想
    嗯,static寫掉了,這是一個靜態(tài)方法,直接通過類名調(diào)用  回復  更多評論
      

    # re: 親密接觸設(shè)計模式(二)-------Singleton(單例) 2007-06-12 15:07 hardson
    synchronized 是多余的  回復  更多評論
      

    # re: 親密接觸設(shè)計模式(二)-------Singleton(單例) 2007-06-18 14:16 想飛就飛
    第二種方式需要synchronized 的吧  回復  更多評論
      

    # re: 親密接觸設(shè)計模式(二)-------Singleton(單例) 2011-03-09 21:03 回歸
    恩,寫掉了,以后注意@一點思想
      回復  更多評論
      

    主站蜘蛛池模板: 777成影片免费观看| 精品无码国产污污污免费| 亚洲精品视频在线播放| 最新猫咪www免费人成| 一级日本高清视频免费观看| 亚洲第一中文字幕| 在线观看特色大片免费视频| 国产成人不卡亚洲精品91| 亚洲av无码潮喷在线观看| 在线免费视频一区| 你懂的在线免费观看| 中文字幕 亚洲 有码 在线| 久久久青草青青国产亚洲免观| 99久久99热精品免费观看国产| 国产精品亚洲精品久久精品| 亚洲ⅴ国产v天堂a无码二区| 国产高清在线精品免费软件| 久久久久成人精品免费播放动漫| 亚洲欧美一区二区三区日产| 久久久亚洲精品国产| 免费一级毛片正在播放| 久久午夜伦鲁片免费无码| 国产精品亚洲а∨天堂2021| 亚洲精品影院久久久久久| 亚洲精品无码久久毛片| 99视频在线精品免费观看6| 97人妻精品全国免费视频| 亚洲av无码专区在线观看亚| 亚洲老熟女@TubeumTV| 亚洲中久无码不卡永久在线观看| 亚洲成在人线aⅴ免费毛片| 鲁丝片一区二区三区免费| 色网站在线免费观看| 亚洲jjzzjjzz在线观看| 亚洲av福利无码无一区二区| 亚洲Av无码国产情品久久| 啦啦啦手机完整免费高清观看| 中文字幕免费观看| 3344在线看片免费| 久久久久久久国产免费看| 四虎精品成人免费视频|