一篇文章教你使用枚舉來實(shí)現(xiàn)java單例模式
傳統(tǒng)的單例寫法解決了什么問題
首先,在大多數(shù)情況下(不包含面試),傳統(tǒng)的單例寫法已經(jīng)完全夠用了。通過 synchronized 關(guān)鍵字解決了多線程并發(fā)使用。
public synchronized static SingleClassV1 getInstance(){ if(instance == null){ instance = new SingleClassV1(); } return instance; }
考慮到每次獲取單例對象都需要加鎖,解鎖。又有人發(fā)明了雙重鎖校驗(yàn) + volatile 關(guān)鍵字模式:
private static volatile SingleClassV2 instance; public static SingletonV2 getInstance() { if(instance == null){ synchronized (SingletonV2.class){ if(instance == null){ instance = new SingletonV2(); } } } return instance; }
另外一種為了解決單例被重復(fù)初始化的寫法:利用類只會(huì)被初始化一次的特性,又有人發(fā)明出來一種內(nèi)部類單例的寫法。
private static class SingletonHolder { private static final SingletonV3 INSTANCE = new SingletonV3(); } public static final SingletonV3 getInstance() { return SingletonHolder.INSTANCE; }
仍然存在的問題
由于 java 中有反射 API 這種變態(tài)的存在,以上所有的私有構(gòu)造方法在反射面前都是毛毛雨。
Class<?> clazzV2 = Class.forName(SingleClassV2.class.getName()); Constructor<?> constructor = clazzV2.getDeclaredConstructors()[0]; constructor.setAccessible(true); Object o = constructor.newInstance();
看來私有方法是防君子不防小人
為什么枚舉就沒有問題
我們來先看一下基于枚舉的單例是什么樣的。
public enum SingleClassV4 { INSTANCE; public String doSomeThing(){ return "hello world"; } }
當(dāng)然,從 java 代碼是看不出來任何端倪的。再使用 javap 看一下字節(jié)碼。
public final class git.frank.SingleClassV4 extends java.lang.Enum<git.frank.SingleClassV4> minor version: 0 major version: 52 flags: ACC_PUBLIC, ACC_FINAL, ACC_SUPER, ACC_ENUM
可以發(fā)現(xiàn),枚舉類型會(huì)幫我們自動(dòng)繼承 java.lang.Enum 類。并且,在 flags 中該類被添加了 ACC_ENUM 標(biāo)識。然后,再看一下枚舉類的構(gòu)造方法:
private git.frank.SingleClassV4(); descriptor: (Ljava/lang/String;I)V flags: ACC_PRIVATE Code: stack=3, locals=3, args_size=3 0: aload_0 1: aload_1 2: iload_2 3: invokespecial #6 // Method java/lang/Enum."<init>":(Ljava/lang/String;I)V 6: return LineNumberTable: line 3: 0//加入Java開發(fā)交流君樣:756584822一起吹水聊天 LocalVariableTable: Start Length Slot Name Signature 0 7 0 this Lgit/frank/SingleClassV4; Signature: #29 // ()V
枚舉類也是要有構(gòu)造方法的,而且也和普通的類沒什么不同,也一樣可以通過反射獲取到:
接下來,讓我們通過反射 invoke 一下他的構(gòu)造方法看看會(huì)發(fā)生什么:
constructor.newInstance();
結(jié)果如下:
Exception in thread "main" java.lang.IllegalArgumentException: Cannot reflectively create enum objects at java.lang.reflect.Constructor.newInstance(Constructor.java:417)
通過看 newInstance 方法代碼的話,就很容易知道原因了:
public T newInstance(Object ... initargs) throws InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {//加入Java開發(fā)交流君樣:756584822一起吹水聊天 ... if ((clazz.getModifiers() & Modifier.ENUM) != 0) throw new IllegalArgumentException("Cannot reflectively create enum objects"); ... T inst = (T) ca.newInstance(initargs); return inst; }
java 的反射 API 在創(chuàng)建對象實(shí)例是判斷了當(dāng)前類是否是枚舉類型,否則就拋異常出來。
總結(jié)
在傳統(tǒng)的單例寫法中,由于私有構(gòu)造方法并不能完全杜絕從外部創(chuàng)建實(shí)例,所以嚴(yán)格來說那些單例的實(shí)現(xiàn)方式是存在漏洞的。
由于 java 的反射 API 已經(jīng)通過寫死的方式限制了不能為枚舉類型創(chuàng)建實(shí)例,所以… 也算了解決了吧。哎呀,這個(gè)東西是也是面試被問到的,正常人誰會(huì)用反射這種外掛去突破單例。
本篇文章就到這里了,希望能給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
IntelliJ IDEA Run時(shí)報(bào)“無效的源發(fā)行版:16“錯(cuò)誤問題及解決方法
這篇文章主要介紹了IntelliJ IDEA Run時(shí)報(bào)“無效的源發(fā)行版:16“錯(cuò)誤問題及解決方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-05-05Java數(shù)據(jù)類型Integer與int的區(qū)別詳細(xì)解析
這篇文章主要介紹了Java數(shù)據(jù)類型Integer與int的區(qū)別詳細(xì)解析,Ingeter是int的包裝類,int的初值為0,Ingeter的初值為null,int和integer(無論new否)比,都為true,因?yàn)闀?huì)把Integer自動(dòng)拆箱為int再去比,需要的朋友可以參考下2023-12-12MyBatisPlus自定義JsonTypeHandler實(shí)現(xiàn)自動(dòng)轉(zhuǎn)化JSON問題
這篇文章主要介紹了MyBatisPlus自定義JsonTypeHandler實(shí)現(xiàn)自動(dòng)轉(zhuǎn)化JSON問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12jmeter+ant+jenkins自動(dòng)化測試環(huán)境配置搭建過程
在搭建jmeter+ant+jenkins環(huán)境有些前提條件,那就是要先配置好java環(huán)境、安裝好jenkins以及配置好jmeter,這樣才能省去很多的事情,對jmeter+ant+jenkins自動(dòng)化測試環(huán)境配置搭建過程感興趣的朋友一起看看吧2021-12-12基于NIO的Netty網(wǎng)絡(luò)框架(詳解)
下面小編就為大家?guī)硪黄贜IO的Netty網(wǎng)絡(luò)框架(詳解)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-06-06Lombok @Slf4j log對象沒有info等方法不可用問題及解決
本文主要介紹了如何解決Spring Boot項(xiàng)目中的日志依賴沖突問題,以及如何使用Lombok和SLF4J進(jìn)行日志記錄,Lombok通過生成Logger對象簡化了日志記錄,而SLF4J提供了一個(gè)統(tǒng)一的日志接口,允許開發(fā)者在運(yùn)行時(shí)選擇不同的日志實(shí)現(xiàn)2024-12-12SpringBoot?pdf打印及預(yù)覽(openhtmltopdf+freemarker)
這篇文章主要介紹了SpringBoot?pdf打印及預(yù)覽(openhtmltopdf+freemarker)2023-05-05