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í)例有沒有被使用到,都會(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-03
Spring實(shí)戰(zhàn)之使用@Resource配置依賴操作示例
這篇文章主要介紹了Spring實(shí)戰(zhàn)之使用@Resource配置依賴操作,結(jié)合實(shí)例形式分析了Spring使用@Resource配置依賴具體步驟、實(shí)現(xiàn)及測(cè)試案例,需要的朋友可以參考下2019-12-12
SpringBoot項(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-04
SpringBoot+MyBatisPlus+MySQL8實(shí)現(xiàn)樹形結(jié)構(gòu)查詢
這篇文章主要為大家詳細(xì)介紹了SpringBoot+MyBatisPlus+MySQL8實(shí)現(xiàn)樹形結(jié)構(gòu)查詢,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-06-06

