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、如何選擇?
-
單例對象占用資源少,不需要延時加載:
枚舉式 好于 餓漢式
-
單例對象占用資源大,需要延時加載:
靜态内部類式 好于 懶漢式
以上為個人了解,有不正确的地方歡迎大家指教