Java 反射修改類的常量值、靜態(tài)變量值、屬性值實(shí)例詳解
前言
有的時(shí)候,我們需要修改一個(gè)變量的值,但變量也許存在于 Jar 包中或其他位置,導(dǎo)致我們不能從代碼層面進(jìn)行修改,于是我們就用到了下面的場景,通過反射來進(jìn)行修改變量的值。
定義一個(gè)實(shí)體類
class Bean{ private static final Integer INT_VALUE = 100; }
利用反射修改私有靜態(tài)常量方法
System.out.println(Bean.INT_VALUE); Field field = Bean.class.getField("INT_VALUE"); //將字段的訪問權(quán)限設(shè)為true:即去除private修飾符的影響 field.setAccessible(true); //去除final修飾符的影響,將字段設(shè)為可修改的 Field modifiersField = Field.class.getDeclaredField("modifiers"); modifiersField.setAccessible(true); modifiersField.setInt(field, field.getModifiers() & ~Modifier.FINAL); //把字段值設(shè)為200 field.set(null, 200); System.out.println(Bean.INT_VALUE);
修改私有靜態(tài)常量測試結(jié)果
100
200
看到測試結(jié)果說明我們的反射修改成功了。
利用反射修改共有靜態(tài)變量方法
class Bean{ public static int nums = 100; } System.out.println(Bean.nums); Field field = Bean.class.getField("nums"); field.set(null, 200); System.out.println(Bean.INT_VALUE);
測試結(jié)果修改成功。
100
200
奇怪的地方
注意到上述代碼的中的靜態(tài)常量類型是Integer,但是我們項(xiàng)目中實(shí)際需要修改的字段類型并不是包裝類型Integer,而是java的基本類型int。
當(dāng)把常量的類型改成int之后。
class Bean{ private static final int INT_VALUE = 100;//把類型由Integer改成了int }
在其他代碼都不變的情況下,代碼輸出的結(jié)果竟然變成了詭異的:
100
100
而且在調(diào)試的過程中發(fā)現(xiàn),在第二次輸出的時(shí)候,內(nèi)存中的Bean.INT_VALUE是已經(jīng)變成了200,但是System.out.println(Bean.INT_VALUE)輸出的結(jié)果卻依然時(shí)詭異的100?!
是反射失效了嗎?
又試了其他幾種類型,發(fā)現(xiàn)這種貌似失效的情會(huì)發(fā)生在int、long、boolean以及String這些基本類型上,而如果把類型改成Integer、Long、Boolean這種包裝類型,或者其他諸如Date、Object都不會(huì)出現(xiàn)失效的情況。
奇怪的原因
對于基本類型的靜態(tài)常量,JAVA在編譯的時(shí)候就會(huì)把代碼中對此常量中引用的地方替換成相應(yīng)常量值。
參考:Modifying final fields in Java
即對于常量 public static final int maxFormatRecordsIndex = 100
if( index > maxFormatRecordsIndex ){ index = maxFormatRecordsIndex ; }
這段代碼在編譯的時(shí)候已經(jīng)被java自動(dòng)優(yōu)化成這樣的:
if( index > 100){ index = 100; }
所以在INT_VALUE是int類型的時(shí)候。
System.out.println(Bean.INT_VALUE);
編譯時(shí)會(huì)被優(yōu)化成下面這樣:
System.out.println(100);
所以,自然,無論怎么修改Boolean.INT_VALUE,System.out.println(Bean.INT_VALUE)都還是會(huì)依然固執(zhí)地輸出100了。
這本身是JVM的優(yōu)化代碼提高運(yùn)行效率的一個(gè)行為,但是就會(huì)導(dǎo)致我們在用反射改變此常量值時(shí)出現(xiàn)類似不生效的錯(cuò)覺。
到此這篇關(guān)于Java 反射修改類的常量值、靜態(tài)變量值、屬性值實(shí)例詳解的文章就介紹到這了,更多相關(guān)Java 反射如何修改類的常量值、靜態(tài)變量值、屬性值內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
快速校驗(yàn)實(shí)體類時(shí),@Valid,@Validated,@NotNull注解無效的解決
這篇文章主要介紹了快速校驗(yàn)實(shí)體類時(shí),@Valid,@Validated,@NotNull注解無效的解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10深入理解JSON及其在Java中的應(yīng)用小結(jié)
json它是一種輕量級的數(shù)據(jù)交換格式,由于其易于閱讀和編寫,同時(shí)也易于機(jī)器解析和生成,因此廣泛應(yīng)用于網(wǎng)絡(luò)數(shù)據(jù)交換和配置文件,這篇文章主要介紹了深入理解JSON及其在Java中的應(yīng)用,需要的朋友可以參考下2023-12-12Java 實(shí)現(xiàn)分布式服務(wù)的調(diào)用鏈跟蹤
分布式服務(wù)中完成某一個(gè)業(yè)務(wù)動(dòng)作,需要服務(wù)之間的相互協(xié)作才能完成,在這一次動(dòng)作引起的多服務(wù)的聯(lián)動(dòng)我們需要用1個(gè)唯一標(biāo)識關(guān)聯(lián)起來,關(guān)聯(lián)起來就是調(diào)用鏈的跟蹤。本文介紹了Java 實(shí)現(xiàn)分布式服務(wù)的調(diào)用鏈跟蹤的步驟2021-06-06利用feign調(diào)用返回object類型轉(zhuǎn)換成實(shí)體
這篇文章主要介紹了利用feign調(diào)用返回object類型轉(zhuǎn)換成實(shí)體,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03SpringCloud微服務(wù)調(diào)用丟失請求頭的問題及解決方案
在Spring Cloud 中微服務(wù)之間的調(diào)用會(huì)用到Feign,但是在默認(rèn)情況下,Feign 調(diào)用遠(yuǎn)程服務(wù)存在Header請求頭丟失問題,下面給大家分享SpringCloud微服務(wù)調(diào)用丟失請求頭的問題及解決方案,感興趣的朋友一起看看吧2024-02-02Java實(shí)現(xiàn)文件監(jiān)控器FileMonitor的實(shí)例代碼
這篇文章主要介紹了Java實(shí)現(xiàn)文件監(jiān)控器FileMonitor的實(shí)例代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-12-12