Java設(shè)計(jì)模式之單例模式Singleton Pattern詳解
1.單例設(shè)計(jì)模式的設(shè)計(jì)思想與應(yīng)用場(chǎng)景
設(shè)計(jì)目的
避免因?yàn)閯?chuàng)建了多個(gè)實(shí)例造成資源的浪費(fèi),且多個(gè)實(shí)例由于多次調(diào)用容易導(dǎo)致結(jié)果出現(xiàn)錯(cuò)誤,而使用單例模式能夠保證整個(gè)應(yīng)用中有且只有一個(gè)實(shí)例。
設(shè)計(jì)思想
(1)不允許其他程序用new對(duì)象: 因?yàn)閚ew就是開(kāi)辟新的空間,在這里更改數(shù)據(jù)只是更改的所創(chuàng)建的對(duì)象的數(shù)據(jù),如果可以new的話(huà),每一次new都產(chǎn)生一個(gè)對(duì)象,這樣肯定保證不了對(duì)象的唯一性。
(2)在該類(lèi)中創(chuàng)建對(duì)象: 因?yàn)椴辉试S其他程序new對(duì)象,所以這里的對(duì)象需要在本類(lèi)中new出來(lái)
(3)對(duì)外提供一個(gè)可以讓其他程序獲取該對(duì)象的方法
應(yīng)用場(chǎng)景
一些常用的工具類(lèi)、線(xiàn)程池、緩存,數(shù)據(jù)庫(kù),數(shù)據(jù)庫(kù)連接池、賬戶(hù)登錄系統(tǒng)、配置文件等程序中可能只允許我們創(chuàng)建一個(gè)對(duì)象。
2.單例模式的寫(xiě)法
單例模式的寫(xiě)法大的方面可以分為5種五種
1.懶漢式
2.餓漢式
3.雙重校驗(yàn)鎖
4.靜態(tài)內(nèi)部類(lèi)
5.枚舉
2.1單例模式的餓漢式
public class Singleton {
private static Singleton instance=new Singleton();
//構(gòu)造器私有化,本身不可創(chuàng)建
private Singleton(){
}
//定義一個(gè)獲取對(duì)象的方法
public static Singleton getInstance(){
return instance;
}
}
餓漢式的靜態(tài)代碼塊寫(xiě)法
public class Singleton{
private static Singleton instance = null;
static{
instance = new Singleton();
}
private Singleton(){
}
public static Singleton getInstance(){
return instance;
}
}
備注:兩種寫(xiě)法的主要區(qū)別在于加載的時(shí)間點(diǎn)上,前者在于靜態(tài)常量顯示初始化之后,后者在與靜態(tài)代碼塊加載運(yùn)行的時(shí)候。
優(yōu)點(diǎn):餓漢式的寫(xiě)法主要的優(yōu)勢(shì)在于實(shí)現(xiàn)方式簡(jiǎn)單,在類(lèi)加載的時(shí)候完成了對(duì)象的實(shí)例化,因此也可以有效避免線(xiàn)程問(wèn)題。
缺點(diǎn):可能造成資源浪費(fèi)的情況,由于對(duì)象實(shí)例是在代碼加載過(guò)程中進(jìn)行的,只要代碼成功加載對(duì)象實(shí)例就會(huì)被創(chuàng)建導(dǎo)致內(nèi)存浪費(fèi)。【被占用的內(nèi)存是非常小的,所以也是推介可以使用的?!?/p>
2.2單例模式的懶漢式
2.2.1 懶漢式普通寫(xiě)法
線(xiàn)程不安全,不推薦使用
public class Singleton {
private static Singleton instance=null;
private Singleton() {};
public static Singleton getInstance(){
if(instance==null){
instance=new Singleton();
}
return instance;
}
}
與餓漢式的寫(xiě)法相比較,懶漢式主要是在對(duì)**getInstance()**方法調(diào)用的時(shí)候進(jìn)行實(shí)例化對(duì)象的操作,既用之再造,節(jié)省了內(nèi)存,故因此稱(chēng)之為懶漢式。
懶漢式的普通寫(xiě)法式存在線(xiàn)程安全問(wèn)題的,總的來(lái)說(shuō)也就是存在重復(fù)創(chuàng)建對(duì)象的風(fēng)險(xiǎn)。假設(shè)線(xiàn)程1進(jìn)來(lái),通過(guò)if(instance==null)的條件判斷準(zhǔn)備運(yùn)行”insatance=new Singleton()“這條代碼前,又有線(xiàn)程2運(yùn)行,此時(shí)還沒(méi)有成功進(jìn)行Singleton對(duì)象實(shí)例的創(chuàng)建,if 的條件判斷仍然成立,線(xiàn)程2也會(huì)去執(zhí)行instance=new Singleton()這條語(yǔ)句,這樣就會(huì)導(dǎo)致多個(gè)Singleton對(duì)象被創(chuàng)建。這就是為什么懶漢式的普通寫(xiě)法會(huì)造成線(xiàn)程安全的原因。
2.2.2懶漢式同步代碼寫(xiě)法
【給創(chuàng)建對(duì)象的語(yǔ)句加鎖】與2.2.1的寫(xiě)法類(lèi)似仍為線(xiàn)程不安全的寫(xiě)法,不推薦使用
public class Singleton7 {
private static Singleton instance=null;
public static Singleton getInstance() {
if (instance == null) {
synchronized (Singleton.class) {
instance = new Singleton();
}
}
return instance;
}
}
與2.2.1相類(lèi)似,synchronized (Singleton.class) 鎖住的只是對(duì)于對(duì)象的創(chuàng)建,仍有有可能再線(xiàn)程1獲得”鑰匙“還未進(jìn)行對(duì)象的創(chuàng)建的時(shí)候,線(xiàn)程二也通過(guò)條件判斷進(jìn)行鎖等待,一旦獲得“鑰匙”也會(huì)進(jìn)行對(duì)象創(chuàng)建,也是會(huì)導(dǎo)致多對(duì)象的創(chuàng)建。
2.2.3懶漢式同步方法寫(xiě)法
線(xiàn)程安全,但效率非常低,不推薦使用
public class Singleton {
private static Singleton instance=null;
private Singleton() {};
public static synchronized Singleton getInstance(){
if(instance==null){
instance=new Singleton();
}
return instance;
}
}
缺點(diǎn):效率太低了,每個(gè)線(xiàn)程在想獲得類(lèi)的實(shí)例時(shí)候,執(zhí)行g(shù)etInstance()方法都要進(jìn)行同步,方法進(jìn)行同步效率太低。
2.2.4單例模式懶漢式雙重校驗(yàn)鎖
推介使用
public class Singleton {
/**
* 懶漢式變種,屬于懶漢式中最好的寫(xiě)法,保證了:延遲加載和線(xiàn)程安全
*/
private static Singleton instance=null;
private Singleton() {};
public static Singleton getInstance(){
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}通過(guò)雙層if(insatance==null)的條件限制,防止了多線(xiàn)程運(yùn)行就算在解鎖條件下多對(duì)象的創(chuàng)建。
通過(guò)synchronized 保證了線(xiàn)程安全,但再第一次對(duì)象成功創(chuàng)建后,就可以通過(guò)return instance 避免了加鎖代碼的重復(fù)運(yùn)行,大大提升了效率。
在線(xiàn)程安全以及效率之間找到了一個(gè)平衡點(diǎn)。
2.3內(nèi)部類(lèi)
public class Singleton{
private Singleton() {};
private static class SingletonHolder{
private static Singleton instance=new Singleton();
}
public static Singleton getInstance(){
return SingletonHolder.instance;
}
}這種方式跟餓漢式方式采用的機(jī)制類(lèi)似,但又有不同。兩者都是采用了類(lèi)裝載的機(jī)制來(lái)保證初始化實(shí)例時(shí)只有一個(gè)線(xiàn)程。不同
的地方在餓漢式方式是只要Singleton類(lèi)被裝載就會(huì)實(shí)例化,沒(méi)有Lazy-Loading的作用,而靜態(tài)內(nèi)部類(lèi)方式在Singleton類(lèi)被裝載時(shí)
并不會(huì)立即實(shí)例化,而是在需要實(shí)例化時(shí),調(diào)用getInstance方法,才會(huì)裝載SingletonHolder類(lèi),從而完成Singleton的實(shí)例化。
類(lèi)的靜態(tài)屬性只會(huì)在第一次加載類(lèi)的時(shí)候初始化,所以在這里,JVM幫助我們保證了線(xiàn)程的安全性,在類(lèi)進(jìn)行初始化時(shí),別的線(xiàn)程是
無(wú)法進(jìn)入的。
優(yōu)點(diǎn):避免了線(xiàn)程不安全,延遲加載,效率高。
2.4枚舉
public enum SingletonEnum {
instance;
private SingletonEnum() {}
public void method(){
}
}也就是直接將枚舉類(lèi)Enum直接當(dāng)作一個(gè)上述例子中的Singleton所對(duì)應(yīng)的實(shí)例去進(jìn)行使用,可以直接通過(guò)SingletonEnum.instance.method();的方式對(duì)實(shí)例中我們要使用的方法進(jìn)行調(diào)用。
不僅能避免多線(xiàn)程同步問(wèn)題,而且還能防止反序列化重新創(chuàng)建新的對(duì)象??赡苁且?yàn)槊杜e在JDK1.5中才添加,所以在實(shí)際項(xiàng)目開(kāi)發(fā)中,很少見(jiàn)人這么寫(xiě)過(guò),這種方式也是最好的一種方式,如果在開(kāi)發(fā)中JDK滿(mǎn)足要求的情況下建議使用這種方式。
總結(jié)
對(duì)于寫(xiě)法的選擇,開(kāi)源項(xiàng)目也是采用的單例模式懶漢式雙重校驗(yàn)鎖這種寫(xiě)法,其實(shí)最安全的寫(xiě)法是枚舉,它的實(shí)現(xiàn)非常簡(jiǎn)單而且最安全可謂很完美,但是可能是因?yàn)橹С值腏DK版本有限又或者是因?yàn)槊杜e大家不熟悉所以目前使用的人并不多,可以更多的進(jìn)行嘗試。
另外當(dāng)我們使用反射機(jī)制時(shí)可能不能保證實(shí)例的唯一性,但是枚舉始終可以保證唯一性【這與枚舉類(lèi)在初始類(lèi)加載的過(guò)程以及進(jìn)行的操作有關(guān)】。
到此這篇關(guān)于Java設(shè)計(jì)模式之單例模式Singleton Pattern詳解的文章就介紹到這了,更多相關(guān)Java單例模式Singleton Pattern內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用SpringBoot_jar方式啟動(dòng)并配置日志文件
這篇文章主要介紹了使用SpringBoot_jar方式啟動(dòng)并配置日志文件操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
Tomcat 實(shí)現(xiàn)WebSocket詳細(xì)介紹
這篇文章主要介紹了Tomcat 如何實(shí)現(xiàn)WebSocket的相關(guān)資料,對(duì)WebSocket協(xié)議通信的過(guò)程進(jìn)行了詳細(xì)介紹,需要的朋友可以參考下2016-12-12
Windows下使用Graalvm將Springboot應(yīng)用編譯成exe大大提高啟動(dòng)和運(yùn)行效率(推薦)
這篇文章主要介紹了Windows下使用Graalvm將Springboot應(yīng)用編譯成exe大大提高啟動(dòng)和運(yùn)行效率,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-02-02
基于Spring中各個(gè)jar包的作用及依賴(lài)(詳解)
下面小編就為大家?guī)?lái)一篇基于Spring中各個(gè)jar包的作用及依賴(lài)(詳解)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11
Java實(shí)現(xiàn)線(xiàn)程同步方法及原理詳解
這篇文章主要介紹了Java實(shí)現(xiàn)線(xiàn)程同步方法及原理詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06
springboot項(xiàng)目中的bootstrap.yml配置不生效的原因及解決(沒(méi)有自動(dòng)提示)
新創(chuàng)建一個(gè) springboot項(xiàng)目,添加了 bootstrap.yml 文件,發(fā)現(xiàn)文件并沒(méi)有如預(yù)期變成綠色葉子,編寫(xiě)的時(shí)候也沒(méi)有自動(dòng)提示,啟動(dòng)的時(shí)候,發(fā)現(xiàn)端口是8080,由此發(fā)現(xiàn)配置并沒(méi)有生效,所以本文給大家講解了springboot項(xiàng)目中的bootstrap.yml配置不生效的原因及解決2024-01-01
基于Spring AOP proxyTargetClass的行為表現(xiàn)總結(jié)
這篇文章主要介紹了Spring AOP proxyTargetClass的行為表現(xiàn)總結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08
讓Java程序自動(dòng)重啟的實(shí)現(xiàn)方法(推薦)
下面小編就為大家?guī)?lái)一篇讓Java程序自動(dòng)重啟的實(shí)現(xiàn)方法(推薦)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03

