前言
本文主要講述單例模式,文中使用通俗易懂的案例,使你更好的學(xué)習(xí)本章知識點(diǎn)并理解原理,做到有道無術(shù)。
一. 什么是單例模式
單例模式是23種設(shè)計(jì)模式中創(chuàng)建型模式的一種,通過單例模式的方法創(chuàng)建的類在當(dāng)前進(jìn)程或者線程中只有一個(gè)實(shí)例。單例模式有兩種比較常見的實(shí)現(xiàn)方式:餓漢式、懶漢式
二. 生活中的單例模式
1.父母
古語有云:身體發(fā)膚,受之父母。咱們每個(gè)人包括小編自己也是,都是父母的愛情結(jié)晶。親生的父母永遠(yuǎn)只會有一對,而你未來的孩子也是一樣,也只有一對親生父母,這就跟單例模式很相似。
2.婚姻
每個(gè)人都渴望擁有能夠完美的愛情,與相愛之人步入婚姻的殿堂,攜手共度余生。而單例模式就像咱們的生活中的婚姻一樣,不管你是否有過婚姻,還是有過多少段婚姻,你的合法妻子最多只會有一個(gè),不可能有多個(gè)?;橐鲋械陌閭H就跟單例模式很相似。
3.人生
每個(gè)人的人生都是不相同的,沒有兩個(gè)一樣的獨(dú)立個(gè)體,畢竟每個(gè)人的生活經(jīng)歷、成長環(huán)境、性格等等都是不一樣的,所以造就了獨(dú)一無二的你。而你就是單例模式最好的表現(xiàn)。
三. 單例模式的實(shí)現(xiàn)
接下來咱們就來講下最基礎(chǔ)的兩種單例模式實(shí)現(xiàn)方式:餓漢式、懶漢式。
1.餓漢式
接下來小編以婚姻中的夫妻對象來舉例,使用餓漢式來實(shí)現(xiàn)。首先來創(chuàng)建一個(gè)餓漢式類和客戶端。
package com.qianfeng.ran
/*
* @author:小編
*
* 由餓漢式創(chuàng)建的類:老婆類
*/
public class Wife{
//創(chuàng)建一個(gè)自身實(shí)例對象的私有化靜態(tài)屬性
private static Wife wife = new Wife();
//創(chuàng)建私有化無參構(gòu)造方法
private Wife(){
}
//創(chuàng)建一個(gè)供外界獲取當(dāng)前實(shí)例對象的公開靜態(tài)方法
public static Wife getInstance(){
return wife;
}
}
class Demo{
public static void main(String[] args) {
Wife wife1 = Wife.getInstance();
Wife wife2 = Wife.getInstance();
//執(zhí)行結(jié)果:
//true
System.out.println(wife1 == wife2);
}
}
從上面案例可以看出,構(gòu)造方法被私有化后,我們只能通過它提供的方法 getInstance() 來獲取當(dāng)前類的實(shí)例對象,而我們每次獲取的對象都會是同一個(gè),所以 Wife 是個(gè)單例模式的類。
而從 Wife 類中我們能看得到它有個(gè)靜態(tài)的私有屬性,這個(gè)私有屬性就是當(dāng)前類的實(shí)例對象。也就是說不管我們還沒有調(diào)用這個(gè)類,類中的靜態(tài)對象屬性在加載進(jìn)內(nèi)存中的時(shí)候已經(jīng)存在了,這個(gè)就是餓漢式的特點(diǎn)。
餓漢式就婚姻里的對象一樣,他(她)一早就出現(xiàn)在這個(gè)世上,只是需要你們彼此尋找到對方而已。
2. 懶漢式
接下來小編再以孩子為例,用懶漢式實(shí)現(xiàn)。創(chuàng)建一個(gè)懶漢式類和客戶端。
package com.qianfeng.ran
/*
* @author:江帥
*
* 由懶漢式創(chuàng)建的類:孩子類
*/
public class Child{
//創(chuàng)建一個(gè)類型為當(dāng)前類的私有化靜態(tài)屬性
private static Child child;
//創(chuàng)建私有化無參構(gòu)造方法
private Child(){
}
//創(chuàng)建一個(gè)供外界獲取當(dāng)前實(shí)例對象的公開靜態(tài)方法
public static Child getInstance(){
//判斷靜態(tài)屬性是否為 null
if(child == null){
//如果為 null 則創(chuàng)建當(dāng)前類對象并賦值給靜態(tài)屬性
child = new Child();
}
return child;
}
}
class Demo{
public static void main(String[] args) {
Child child1 = Child.getInstance();
Child child2 = Child.getInstance();
//執(zhí)行結(jié)果:
//true
System.out.println(child1 == child2);
}
}
這個(gè)案例也是一樣構(gòu)造方法被私有化,只能通過其提供的靜態(tài)方法獲取其實(shí)例對象。
在 Child 類中它也有個(gè)靜態(tài)的私有屬性,但是這個(gè)靜態(tài)屬性并沒有賦值,只有在第一次調(diào)用 Child 類獲取實(shí)例方法 getInstance() 的時(shí)候,才會創(chuàng)建當(dāng)前類的實(shí)例對象并賦值給其屬性并返回。因此在沒有調(diào)用其方法之前,對象是沒有創(chuàng)建的,只有等到需要的時(shí)候才會被創(chuàng)建,這就是懶漢式。
懶漢式就孩子一樣,他們并不會一開始就存在,只有當(dāng)你創(chuàng)造他的時(shí)候才會存在,而你與你的對象創(chuàng)建出來每一胎的孩子都會是獨(dú)一無二的,即使你多生幾個(gè),每一個(gè)都是唯一的。
四. 總結(jié)
單例模式就像生活中獨(dú)一無二的事物一樣,只不過有些早已存在,而有些需要你去創(chuàng)建。在程序中很多場景都會運(yùn)用到單例模式,它能減少我們需要的對象,降低服務(wù)器的壓力,提高運(yùn)行效率。
不過單例模式的兩種模式中,懶漢式可能會出現(xiàn)線程安全問題。就像一胎生2個(gè)孩子,這樣每一胎就是獨(dú)立的一個(gè)對象了。