java使用單例設計模式的方式有很多種,比如餓漢式,懶漢式寸宵,靜態(tài)內(nèi)部類式又谋,雙重檢測鎖式以及枚舉方式靡努,這里主要講枚舉式
一突琳、案例
1键菱、新建一個接口類
/**
* @author Gjing
**/
public interface MySingleton {
void doSomething();
}
2顺饮、新建枚舉類火焰,實現(xiàn)上面的接口
/**
* @author Gjing
**/
public enum Singleton implements MySingleton{
/**
* 實例
*/
INSTANCE{
@Override
public void doSomething() {
System.out.println("執(zhí)行方法劲装。。昌简。");
}
}
}
3占业、測試
/**
* @author Gjing
**/
public class Test {
public static void main(String[] args) {
Singleton.INSTANCE.doSomething();
}
}
以上為簡單的使用枚舉方式實現(xiàn)單例模式
二、知識點
1纯赎、枚舉方式實現(xiàn)單例好處
線程安全谦疾、調(diào)用效率高,但不能延時加載犬金,枚舉本身就是單例模式念恍。由JVM從根本上提供保障!避免通過反射和反序列化的漏洞佑附!
2樊诺、以上四種方式的比較
- 餓漢式:
餓漢式單例模式代碼中,static變量會在類裝載時初始化音同,此時也不會涉及多個線程對象訪問該對象的問題词爬。虛擬機保證只會裝載一次該類,肯定不會發(fā)生并發(fā)訪問的問題权均。
因此顿膨,可以省略synchronized關鍵字。問題:如果只是加載本類叽赊,而不是調(diào)用getInstance()恋沃,甚至永遠沒有調(diào)用,則會造成資源浪費必指!
- 懶漢式(單例對象延時加載):
要點:lazy load囊咏! 延時加載,懶加載塔橡!真正用的時候才加載梅割!
問題:資源利用率高了。但是葛家,每次調(diào)用getInstance()方法都要同步户辞,并發(fā)效率低。
- 靜態(tài)內(nèi)部類實現(xiàn)方式(也是一種懶加載方式):
要點:外部類沒有static屬性癞谒,則不會像餓漢式那樣立即加載對象底燎。
只有真正調(diào)用getInstance()刃榨,才會加載靜態(tài)內(nèi)部類。加載類時是線程安全的双仍。instance是static final類型枢希,保證了內(nèi)存中只有這樣一個實例存在,而且只能被賦值一次朱沃,從而保證了線程安全性兼?zhèn)淞瞬l(fā)高效調(diào)用和延遲加載的優(yōu)勢晴玖!
- 雙重檢測鎖式
由于JVM底層內(nèi)部模型原因,偶爾會出問題为流。不建議使用
- 枚舉式(推薦):
線程安全、調(diào)用效率高让簿,但不能延時加載敬察,并且可以天然的防止反射和反序列化漏洞!
由上可得知:
1. 餓漢式
線程安全尔当,調(diào)用效率高 莲祸,但是不能延遲加載
2. 懶漢式
線程安全,調(diào)用效率不高椭迎,能延遲加載
3. 雙重檢測鎖式
由于JVM底層內(nèi)部模型原因锐帜,偶爾會出問題。不建議使用
4. 靜態(tài)內(nèi)部類式
線程安全畜号,資源利用率高缴阎,可以延時加載
5. 枚舉單例
線程安全,調(diào)用效率高简软,但是不能延遲加載
3蛮拔、如何選擇?
單例對象占用資源少痹升,不需要延時加載:
枚舉式 好于 餓漢式單例對象占用資源大建炫,需要延時加載:
靜態(tài)內(nèi)部類式 好于 懶漢式
以上為個人理解,有不正確的地方歡迎大家指教