溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java中反射修改類的常量值、靜態(tài)變量值

發(fā)布時(shí)間:2021-01-18 14:47:05 來(lái)源:億速云 閱讀:468 作者:Leah 欄目:開(kāi)發(fā)技術(shù)

今天就跟大家聊聊有關(guān)Java中反射修改類的常量值、靜態(tài)變量值,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

定義一個(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"); 
//將字段的訪問(wèn)權(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)常量測(cè)試結(jié)果

100
200

看到測(cè)試結(jié)果說(shuō)明我們的反射修改成功了。

利用反射修改共有靜態(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);

測(cè)試結(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)試的過(guò)程中發(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)失效的情況。

奇怪的原因

對(duì)于基本類型的靜態(tài)常量,JAVA在編譯的時(shí)候就會(huì)把代碼中對(duì)此常量中引用的地方替換成相應(yīng)常量值。

參考:Modifying final fields in Java

即對(duì)于常量 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);

所以,自然,無(wú)論怎么修改Boolean.INT_VALUE,System.out.println(Bean.INT_VALUE)都還是會(huì)依然固執(zhí)地輸出100了。

看完上述內(nèi)容,你們對(duì)Java中反射修改類的常量值、靜態(tài)變量值有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI