問題描述
如何將非字符串常量轉為字符串常量?toString
和 String.valueOf
貌似都無法編譯。
在給注解屬性賦值的時候:
private final static String MAX_VALUE_AS_STRING = Long.toString(Long.MAX_VALUE);
...
@Annotation(value = MAX_VALUE_AS_STRING)
編譯器一直報錯:
attribute must be constant 值類型必須是常量
MAX_VALUE_AS_STRING
不是明明就是個常量,而且是運行時常量啊。而且在官方的文檔中,明明就有 Integer.MAX_VALUE / 2
的說明。
為了搞清楚這一點,還是要從官方文檔入手,細細分析什么是常量。
## Java編譯時常量解釋
首先,編譯器告訴了你這個屬性必須是個常量值,但是并沒有告訴你,這個屬性其實必須是一個編譯時常量值。所謂編譯時常量就是在編譯階段已經可以確定其值的常量。
Oracle的官方文檔 15.28 Constant Expressions 也詳細的解釋了,什么是 ** Compile-time Constant **
1. 原始類型字面量,或者String字面量
2. 能轉型為原始類型字面量,或String字面量的常量
3. 一元運算符(+,-,~,!,但不包含++, --) 和1,2組成的表達式
4. 多元運算符(*,/和%)和1,2組成的表達式
5. 附加運算符( additive operators) (+ 或 -)與之前幾條組成的表達式
6. 位移運算符(<<,>>, >>>)和之前幾條組成的表達式
7. 關系運算符(<,<=,>,>= ,不包括 instanceof)與之前幾條組成的表達式
8. 關系運算符(==,!=)與之前幾條組成的表達式
9. 位運算符(&, ^, |)與之前幾條組成的表達式
10. 條件與和條件或運算符(&&, ||) 與之前幾條組成的表達式
11. 三元運算符 (?:)和之前幾條組成的表達式
12. 帶括號的表達式,括號內也是常量表達式
13. 引用常量變量的簡單變量 [§6.5.6.1](https://docs.oracle.com/javase/specs/jls/se7/html/jls-6.html#jls-6.5.6.1)
14. 類中的常量變量引用,使用類的全限定名或類名進行引用(String.class)
由上述14條規則組成的常量成為編譯時常量。
所以針對上一個問題,任何的方法調用其實都不在描述范圍當中。因此不管是使用 Long.toString(Long.MAX_VALUE)
還是使用 String.valueOf(Long.MAX_VALUE)
,在編譯時都是非法的。
解決方案
既然不能使用方法調用,進行類型轉換,如何實現將 Long.MAX_VALUE 轉換為常量字符串呢?
其實很簡單:
public static final String MAX_VALUE_AS_STRING = Long.MAX_VALUE + "";
這個方式就符合原始類型字面量和String字面量的二元運算這條規則。
終于搞明白了:)