詳解java中的6種單例寫法及優(yōu)缺點(diǎn)
在java中,單例有很多種寫法,面試時(shí),手寫代碼環(huán)節(jié),除了寫算法題,有時(shí)候也會(huì)讓手寫單例模式,這里記錄一下單例的幾種寫法和優(yōu)缺點(diǎn)。
1.初級(jí)寫法
2.加鎖
3.餓漢式
4.懶漢式
5.雙鎖檢驗(yàn)
6.內(nèi)部類
1.初級(jí)寫法
package com.java4all.test6;
/**
* Author: yunqing
* Date: 2018/8/13
* Description:單例模式 -- 初級(jí)
*/
public class Singleton {
private static Singleton singleton = null;
public Singleton() {
}
/**并發(fā)下會(huì)產(chǎn)生多個(gè)實(shí)例*/
public static Singleton getInstance(){
if(singleton == null){
singleton = new Singleton();
}
return singleton;
}
}
上面這種寫法,在并發(fā)環(huán)境下,會(huì)出現(xiàn)多個(gè)實(shí)例。
2.加鎖
我們優(yōu)化上面的代碼,遇到并發(fā),很容易想到加鎖,把獲取對(duì)象的方法加上關(guān)鍵字synchronized,很巧,這種寫法也稱為懶漢式單例 ,如下:
package com.java4all.test6;
/**
* Author: yunqing
* Date: 2018/8/13
* Description:
*/
public class BadSynchronizedSingleton {
private static BadSynchronizedSingleton synchronizedSingleton;
private BadSynchronizedSingleton() {
}
/**
* 缺點(diǎn):等待時(shí)間長(zhǎng)
* 這種整個(gè)方法都同步了,效率很低
* */
public synchronized static BadSynchronizedSingleton getInstance(){
if(synchronizedSingleton == null){
synchronizedSingleton = new BadSynchronizedSingleton();
}
return synchronizedSingleton;
}
}
但是,顯然,我們把整個(gè)方法都同步了,效率很低下,我們可以繼續(xù)優(yōu)化,只在創(chuàng)建實(shí)例的地方加上同步,參考5雙鎖檢驗(yàn)。
3.餓漢式
餓漢式的特點(diǎn)是:類在加載時(shí)就直接初始化了實(shí)例。即使沒(méi)用到,也會(huì)實(shí)例化。
package com.java4all.test6;
/**
* Author: yunqing
* Date: 2018/8/13
* Description:餓漢式單例模式--類初始化時(shí)就自行實(shí)例化
*/
public class ESingleton {
/**類在加載的時(shí)候直接進(jìn)行初始化*/
private static final ESingleton ESINGLETON = new ESingleton();
private ESingleton() {}
/**對(duì)外暴露唯一接口 提供單例對(duì)象*/
public static ESingleton geteSingleton(){
return ESINGLETON;
}
}
4.懶漢式
懶漢式的特點(diǎn)是:用到這個(gè)實(shí)例時(shí)才去調(diào)用方法實(shí)例化。這個(gè)和2中的看起來(lái)是一樣的,因?yàn)檫@個(gè)實(shí)例化方法加了synchronized ,這樣安全一些。
package com.java4all.test6;
/**
* Author: yunqing
* Date: 2018/8/13
* Description:懶漢式單例模式
* 第一次調(diào)用時(shí)實(shí)例化
*/
public class BSingleton {
private static BSingleton bSingleton ;
private BSingleton() {
}
/**
* 整個(gè)方法鎖住了,效率較低
* @return
*/
public synchronized static BSingleton getbSingleton(){
if(bSingleton == null){
bSingleton = new BSingleton();
}
return bSingleton;
}
}
5.雙鎖檢驗(yàn)
雙重非空判斷,new對(duì)象前加一次鎖。
volatile關(guān)鍵字,考慮的是,new關(guān)鍵字在虛擬機(jī)中執(zhí)行時(shí)其實(shí)分為很多步驟,具體原因可以參考深入理解java虛擬機(jī)一書(考慮的是這個(gè)new關(guān)鍵字字節(jié)碼執(zhí)行時(shí)是非原子性的),而volatile關(guān)鍵字可以防止指令重排。
package com.java4all.test6;
/**
* Author: yunqing
* Date: 2018/8/13
* Description:雙鎖檢驗(yàn)
*/
public class SynchronizedSingleton {
/**volatile防止指令重排*/
private static volatile SynchronizedSingleton singleton;
private SynchronizedSingleton() {
}
/**只是在實(shí)例為空時(shí)才進(jìn)行同步創(chuàng)建
* 為什么做了2次判斷?
* A線程和B線程同時(shí)進(jìn)入同步方法0
* 然后都在1位置處判斷了實(shí)例為null
* 然后都進(jìn)入了同步塊2中
* 然后A線程優(yōu)先進(jìn)入了同步代碼塊2中(B線程也進(jìn)入了),然后創(chuàng)建了實(shí)例
* 此時(shí),如果沒(méi)有3處的判斷,那么A線程創(chuàng)建實(shí)例同時(shí),B線程也會(huì)創(chuàng)建一個(gè)實(shí)例
* 所以,還需要做2次判斷
* */
public static SynchronizedSingleton getInstance(){//0
if(singleton == null){//1
synchronized (SynchronizedSingleton.class){//2
if(singleton == null){//3
singleton = new SynchronizedSingleton();//4
}
}
}
return singleton;
}
}
6.內(nèi)部類
package com.java4all.test6;
/**
* Author: yunqing
* Date: 2018/9/15
* Description:靜態(tài)內(nèi)部類
* 優(yōu)點(diǎn):1.延遲,外部類初始化時(shí)不會(huì)進(jìn)行實(shí)例創(chuàng)建,要用時(shí)才會(huì)創(chuàng)建
* 2.安全,靜態(tài)成員變量
*/
public class FSingleton {
private FSingleton() {
}
public static FSingleton getInstance(){
return Singleton.fSingleton;
}
private static class Singleton{
private static FSingleton fSingleton = new FSingleton();
}
}
總結(jié)
以上所述是小編給大家介紹的java中的6種單例寫法,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
解決使用ProcessBuilder踩到的坑及注意事項(xiàng)
這篇文章主要介紹了解決使用ProcessBuilder踩到的坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06
MybatisPlus 主鍵策略之type=IdType.ASSIGN_ID等詳解
雪花算法(雪花)是微博開源的分布式ID生成算法其核心思想就是:使用一個(gè)64位的長(zhǎng)型的數(shù)字作為全局唯一ID,這篇文章主要介紹了MybatisPlus 主鍵策略(type=IdType.ASSIGN_ID等詳解),需要的朋友可以參考下2024-04-04
Springboot如何設(shè)置靜態(tài)資源緩存一年
這篇文章主要介紹了Springboot如何設(shè)置靜態(tài)資源緩存一年,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11
SpringBoot解決循環(huán)調(diào)用問(wèn)題
作者在將SpringBoot從1.5版本升級(jí)至2.6版本,并遷移至阿里云上運(yùn)行后,遇到了循環(huán)調(diào)用問(wèn)題,在Jetty容器中運(yùn)行沒(méi)問(wèn)題,但在Tomcat容器中就出現(xiàn)了循環(huán)引用問(wèn)題,原因是SpringBoot 2.6不鼓勵(lì)循環(huán)引用,暴露出該問(wèn)題,作者提供了兩種解決思路2024-10-10
java實(shí)現(xiàn)XML增加元素操作簡(jiǎn)單示例
這篇文章主要介紹了java實(shí)現(xiàn)XML增加元素操作,結(jié)合簡(jiǎn)單實(shí)例形式分析了java針對(duì)xml格式數(shù)據(jù)的讀取、遍歷、創(chuàng)建等操作技巧,需要的朋友可以參考下2017-02-02
Spring Security OAuth2認(rèn)證授權(quán)示例詳解
這篇文章主要介紹了Spring Security OAuth2認(rèn)證授權(quán)示例詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09
一次 Java 內(nèi)存泄漏的排查解決過(guò)程詳解
這篇文章主要介紹了一次 Java 內(nèi)存泄漏的排查過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-07-07

