天天看點

java單例模式

java使用單例設計模式的方式有很多種,比如餓漢式,懶漢式,靜态内部類式,雙重檢測鎖式以及枚舉方式,這裡主要講枚舉式。

一、案例

1、建立一個接口類

/**
 * @author Gjing
 **/
public interface MySingleton {
    void doSomething();
}           

2、建立枚舉類,實作上面的接口

/**
 * @author Gjing
 **/
public enum  Singleton implements MySingleton{
    /**
     * 執行個體
     */
    INSTANCE{
        @Override
        public void doSomething() {
            System.out.println("執行方法。。。");
        }
    }
}           

3、測試

/**
 * @author Gjing
 **/
public class Test {
    public static void main(String[] args) {
        Singleton.INSTANCE.doSomething();
    }
}           

以上為簡單的使用枚舉方式實作單例模式

二、擴充

1、枚舉方式實作單例的好處

線程安全、調用效率高,但不能延時加載,枚舉本身就是單例模式。由JVM從根本上提供保障!避免通過反射和反序列化的漏洞!

2、四種方式的比較

  • 餓漢式:

餓漢式單例模式代碼中,static變量會在類裝載時初始化,此時也不會涉及多個線程對象通路該對象的問題。虛拟機保證隻會裝載一次該類,肯定不會發生并發通路的問題。

是以,可以省略synchronized關鍵字。問題:如果隻是加載本類,而不是調用getInstance(),甚至永遠沒有調用,則會造成資源浪費!

  • 懶漢式(單例對象延時加載):

要點:lazy load! 延時加載,懶加載!真正用的時候才加載!

問題:資源使用率高了。但是,每次調用getInstance()方法都要同步,并發效率低。

  • 靜态内部類實作方式(也是一種懶加載方式):

要點:外部類沒有static屬性,則不會像餓漢式那樣立即加載對象。

隻有真正調用getInstance(),才會加載靜态内部類。加載類時是線程安全的。instance是static final類型,保證了記憶體中隻有這樣一個執行個體存在,而且隻能被指派一次,進而保證了線程安全性兼備了并發高效調用和延遲加載的優勢!

  • 雙重檢測鎖式
由于JVM底層内部模型原因,偶爾會出問題。不建議使用
  • 枚舉式(推薦):
線程安全、調用效率高,但不能延時加載,并且可以天然的防止反射和反序列化漏洞!

3、總結:

1. 餓漢式
 
 線程安全,調用效率高 ,但是不能延遲加載
 
 2. 懶漢式
 
 線程安全,調用效率不高,能延遲加載
 
 3. 雙重檢測鎖式
 
 由于JVM底層内部模型原因,偶爾會出問題。不建議使用
 
 4. 靜态内部類式
 
 線程安全,資源使用率高,可以延時加載
 
 5. 枚舉單例
 
 線程安全,調用效率高,但是不能延遲加載           

4、如何選擇?

  1. 單例對象占用資源少,不需要延時加載:

    枚舉式 好于 餓漢式

  2. 單例對象占用資源大,需要延時加載:

    靜态内部類式 好于 懶漢式

以上為個人了解,有不正确的地方歡迎大家指教