
本文介绍了Java 反射修改非静态final字段值的正确方法。由于Java版本更新带来的安全限制传统的修改modifiers字段的方法已经失效。本文将提供基于Varhandle的解决方案并详细说明所需的JVM启动参数以帮助开发者在必要时绕过这些限制。在Java中final关键字用于声明一个不可变的变量。这意味着一旦变量初始化其值就无法修改。然而在某些特殊情况下我们可能需要通过反射来修改final字段的值。在Java 直接修改Field对象的modifiers字段在12和更高版本中不再有效因为Java的安全机制得到了加强。本文将介绍Java中的一个字段 17中仍然可行的解决方案。用VarHandle修改final字段Java 9引入了VarHandle它提供了一种更安全、更灵活的访问和操作变量的方式包括final字段。如何使用VarHandle在Java 17修改final字段的步骤获取VarHandle对象 使用MethodHandles.privateLookupIn()方法获取具有足够权限的Lookup对象然后使用该对象在Field类中搜索modifiers字段的VarHandle。移除FINAL修饰符 使用VarHandlel.set()方法将modifiers字段的值更新为原始值和modifiers.FINAL取反后的结果。以下是示例代码立即学习“Java免费学习笔记(深入)import java.lang.invoke.MethodHandles; import java.lang.invoke.VarHandle; import java.lang.reflect.Field; import java.lang.reflect.Modifier; class Foo { private final String bar; public Foo(String bar) { this.bar bar; } public String getBar() { return this.bar; } } public class Example { public static void main(String[] args) throws Throwable { Foo foo new Foo(foobar); System.out.println(foo.getBar()); Field field foo.getClass().getDeclaredField(bar); field.setAccessible(true); VarHandle MODIFIERS; var lookup MethodHandles.privateLookupIn(Field.class, MethodHandles.lookup()); MODIFIERS lookup.findVarHandle(Field.class, modifiers, int.class); MODIFIERS.set(field, field.getModifiers() ~Modifier.FINAL); field.set(foo, new value); // 设置新的值 System.out.println(foo.getBar()); } }JVM启动参数为了使上述代码成功运行在启动JVM时需要添加以下参数--add-opensjava.base/java.lang.reflectALL-UNNAMED --add-opensjava.base/java.langALL-UNNAMED该参数允许代码访问java.lang.reflect和java.使用VarHandle所需的lang包中的私人成员。ALL-UNNAMED表示允许所有未命名模块访问这些包。注意事项安全风险 使用反射修改final字段的值可能会破坏程序的预期行为并导致难以调试的错误。小心使用此技术并确保充分了解其潜在影响。模块化 在模块化环境中需要确保目标类所在的模块允许反射访问。如果模块不导出包含目标类的包则需要使用--add-显式允许opens参数访问。性能 反射操作通常比直接访问字段慢。因此应尽量避免在性能敏感的代码中使用反射来修改final字段。版本兼容性 虽然这种方法在Java 17有效但未来的Java版本可能会引入新的安全限制导致该方法失效。升级Java版本后请务必测试代码以确保其仍能正常工作。总结本文介绍了一种Java VarHandle用于修改final字段的值。这种方法虽然可以绕过Java的安全限制但要谨慎使用充分了解其潜在风险。在大多数情况下更好的解决方案是重新设计代码以避免需要修改final字段的值。只有在确实必要的情况下才应该考虑使用反射。