寫這篇文章的起因是前幾天同事改了一個常量類中的提示,發(fā)布到測試環(huán)境后沒有生效,正好看 《Java 解惑(謎題 93: 類的戰(zhàn)爭)》 提到了這個問題,所以寫篇文章記錄一下。
以下均使用命令行進行演示,至于為什么沒有使用 IDE 后面會提到。
先看一個簡單的 Constants 類:
/**
* Created by Poison on 15/05/2017.
*/
public class Constants {
public static final String a = "before fixing";
}
再看下 Solution 類:
/**
* Created by Poison on 15/05/2017.
*/
public class Solution {
public static void main(String[] args) {
System.out.println(Constants.a);
}
}
編譯,運行 Solution 的主函數(shù),毫無疑問結(jié)果如圖:
before fixing
現(xiàn)在我們把 Constants 類修改為:
/**
* Created by Poison on 15/05/2017.
*/
public class Constants {
public static final String a = "after fixing";
}
重新編譯 Constants 類,再運行 Solution 的主函數(shù),輸出結(jié)果如圖:
after fixing
為什么修改沒有生效?是 Constants 類的問題,還是 Solution 類的問題?
我們先反編譯 Constants 類看看:
javap Constants
由上圖可見,對 Constants 類的修改是生效的。
再看反編譯的 Solution 類:
javap Solution
看到這里,原因也就明確了,常量變量會被編譯進那些引用它們的類中。這和筆者同事前幾日遇到的情況一模一樣,同事在本地開發(fā)時修改了常量類中的常量字段的值,本地是生效的,原因是因為本地開發(fā)使用了 IDE,而 IDE 將引用到常量類的類也重新編譯了,所以能看到最新的值,而在發(fā)布到測試環(huán)境的過程中,打包機僅僅將常量類所屬的模塊進行了重新編譯,未將引用常量的類的模塊重新編譯,所以當時看見的是更改前的值,同事將常量類的 class 文件反編譯后看見的也是修改后的值,但是卻忘了看引用該常量類的類,所以當時沒有發(fā)現(xiàn)這個問題。
Java 解惑
Java 虛擬機規(guī)范(Java SE 8版)
更多文章,請訪問 田爽Poison