2012-12-31 45 views
5

我最近偶然发现了Change private static final field using Java reflection和测试的polygenelubricants'EverythingIsTrue类,工作正常,System.out.format("Everything is %s", false);打印Everything is true确实。但是,当我改变代码为无法使用java反射更改静态最终字段?

public class EverythingIsTrue { 

    public static final boolean FALSE = false; 

    static void setFinalStatic(Field field, Object newValue) throws Exception { 
     field.setAccessible(true); 
     Field modifiersField = Field.class.getDeclaredField("modifiers"); 
     modifiersField.setAccessible(true); 
     modifiersField.setInt(field, field.getModifiers() & ~Modifier.FINAL); 
     field.set(null, newValue); 
    } 

    public static void main(String[] args) throws Exception { 
     setFinalStatic(EverythingIsTrue.class.getField("FALSE"), true); 
     System.out.format("Everything is %s", FALSE); 
    } 
} 

它打印

Everything is false 

有谁知道为什么吗? setFinalStatic实际上是否工作?

回答

13

当访问原始静态final字段时,Java编译器将假定该值是一个常量并且嵌入该值,而不是生成访问该字段的代码。这意味着编译器将替换为FALSE字段的参考,其值为false。如果使用反射来访问该字段,则会看到该字段的值实际上已更改。

这不适用于非原始字段,因为对象引用的值在编译时不能内联。

+1

这是完全一样的答案,因为我的 - 只能用更多的话! – dty

+0

@dty它看起来像是无限的更多的单词;) –

+0

我删除了我的答案,因为你的答案已经被接受了,而我的答案并没有增加任何价值 - 只是简洁! ;-) – dty

17

通过调用方法调用的结果可以避免编译器内联,即使是虚拟调用也是如此。

public class Main { 
    // value is not known at compile time, so not inlined 
    public static final boolean FLAG = Boolean.parseBoolean("false"); 

    static void setFinalStatic(Class clazz, String fieldName, Object newValue) throws NoSuchFieldException, IllegalAccessException { 
     Field field = clazz.getDeclaredField(fieldName); 
     field.setAccessible(true); 
     Field modifiers = field.getClass().getDeclaredField("modifiers"); 
     modifiers.setAccessible(true); 
     modifiers.setInt(field, field.getModifiers() & ~Modifier.FINAL); 
     field.set(null, newValue); 
    } 

    public static void main(String... args) throws Exception { 
     System.out.printf("Everything is %s%n", FLAG); 
     setFinalStatic(Main.class, "FLAG", true); 
     System.out.printf("Everything is %s%n", FLAG); 
    } 
} 

打印

Everything is false 
Everything is true 
+0

这太棒了!使用通用方法,您可以为非基元构造虚拟方法'static T placeHolder(){return null; }' – flakes

+0

@ mc-emperor,以前的版本更具可读性。 – dit

+0

@dit它是控制台输出,所以它不应该被格式化,就好像它是源代码。 –

相关问题