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