Java單例的寫法詳解
單例模式,顧名思義,就是全局只保存有一個(gè)實(shí)例并且能夠避免用戶去手動(dòng)實(shí)例化,所以單例模式的各種寫法都有一個(gè)共同點(diǎn),不能通過(guò)new關(guān)鍵字去創(chuàng)建對(duì)象,因此,如果能夠通過(guò)構(gòu)造方法實(shí)例化,那么就一定要將其聲明為私有。
餓漢式
public class PersonResource { public static final PersonResource PERSON_RESOURCE_SINGLETON = new PersonResource(); private PersonResource(){} public static PersonResource getInstance() { return PERSON_RESOURCE_SINGLETON; } }
這種方式可以說(shuō)是最安全,也最簡(jiǎn)單的了,但卻有一個(gè)缺點(diǎn),那就是無(wú)論這個(gè)實(shí)例有沒(méi)有被使用到,都會(huì)被實(shí)例化,頗有些浪費(fèi)資源
懶漢式一
既然前一種方法有些浪費(fèi)資源,那就換一種寫法,讓類在被調(diào)用的時(shí)候?qū)嵗?/p>
public class PersonResource { private static PersonResource personResourceSingleton; private PersonResource() { } public static PersonResource getPersonResourceSingleton(){ if(null==personResourceSingleton){ personResourceSingleton = new PersonResource(); } return personResourceSingleton; } }
這種方式能夠在需要用到該實(shí)例的時(shí)候再初始化,也能夠在單線程下很好的運(yùn)行,但如果是多線程就容易出現(xiàn)問(wèn)題了。
懶漢式二
public class PersonResource { private static PersonResource personResourceSingleton; private PersonResource() { } public static PersonResource getPersonResourceSingleton(){ if(null==personResourceSingleton){ personResourceSingleton = new PersonResource(); } return personResourceSingleton; } }
多線程之所以會(huì)出現(xiàn)問(wèn)題,是因?yàn)槎鄠€(gè)線程能夠并發(fā)執(zhí)行g(shù)etPersonResourceSingleton方法,從而導(dǎo)致在判斷是否為空時(shí)出現(xiàn)問(wèn)題。
既然如此,加上鎖 ,使其互斥即可。這里又出現(xiàn)了一個(gè)問(wèn)題,每次獲取實(shí)例的時(shí)候都需要加鎖解鎖,而當(dāng)一個(gè)實(shí)例已經(jīng)被產(chǎn)生后,再加鎖就有些多余了;
懶漢式三(雙重檢查)
public class PersonResource { private PersonResource(){ } private volatile static PersonResource personResource; public static PersonResource getInstance(){ if(personResource==null){ synchronized (PersonResource.class){ if(personResource==null){ personResource = new PersonResource(); } } } return personResource; } }
既然實(shí)例確定產(chǎn)生后不再需要加鎖,那我們?cè)讷@取鎖前先判斷一次是否已經(jīng)有實(shí)例存在就可以解決問(wèn)題了
靜態(tài)內(nèi)部類
public class PersonResource { private PersonResource(){} private static class PersonResourceHolder{ public static PersonResource personResourceSingleton = new PersonResource(); } public static PersonResource getInstance(){ return PersonResourceHolder.personResourceSingleton; } }
除了雙重檢查能夠保證安全的單例外,用一個(gè)靜態(tài)內(nèi)部類去持有單例也是可以的,靜態(tài)內(nèi)部類保證了不會(huì)隨外部類的加載而加載,這保證了延遲加載,同時(shí)在加載該類的時(shí)候就實(shí)例化單例,保證了線程安全;
枚舉
public enum PersonResource { /** * PersonResource單例 */ personResource; public void setPersonResource(){ } }
以上幾種方式基本就夠用了,但都有一個(gè)共同的缺點(diǎn),面對(duì)序列化和反序列化,是無(wú)法保證單例的,但枚舉的特性卻能保證這一點(diǎn)
總結(jié)
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Java動(dòng)態(tài)顯示文件上傳進(jìn)度實(shí)現(xiàn)代碼
這篇文章主要為大家詳細(xì)介紹了Java動(dòng)態(tài)顯示文件上傳進(jìn)度實(shí)現(xiàn)代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01使用Mybatis接收Integer參數(shù)的問(wèn)題
這篇文章主要介紹了使用Mybatis接收Integer參數(shù)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03Spring實(shí)戰(zhàn)之使用@Resource配置依賴操作示例
這篇文章主要介紹了Spring實(shí)戰(zhàn)之使用@Resource配置依賴操作,結(jié)合實(shí)例形式分析了Spring使用@Resource配置依賴具體步驟、實(shí)現(xiàn)及測(cè)試案例,需要的朋友可以參考下2019-12-12SpringBoot項(xiàng)目運(yùn)行jar包啟動(dòng)的步驟流程解析
這篇文章主要介紹了SpringBoot項(xiàng)目運(yùn)行jar包啟動(dòng)的步驟流程,本文分步驟通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2020-07-07詳解Springboot之Logback的使用學(xué)習(xí)
Logback是SpringBoot內(nèi)置的日志處理框架,你會(huì)發(fā)現(xiàn)spring-boot-starter其中包含了spring-boot-starter-logging,該依賴內(nèi)容就是Spring Boot默認(rèn)的日志框架logback,本文詳細(xì)介紹了該框架 ,需要的朋友可以參考下2021-05-05在Spring Boot中如何使用數(shù)據(jù)緩存
本篇文章主要介紹了在Spring Boot中如何使用數(shù)據(jù)緩存,具有一定的參考價(jià)值,有興趣的可以了解一下。2017-04-04SpringBoot+MyBatisPlus+MySQL8實(shí)現(xiàn)樹形結(jié)構(gòu)查詢
這篇文章主要為大家詳細(xì)介紹了SpringBoot+MyBatisPlus+MySQL8實(shí)現(xiàn)樹形結(jié)構(gòu)查詢,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-06-06