版权声明:本文源自简书tianma,转载请务必注明出处: //www.greatytc.com/p/2d490b0155ad
之前在阅读其他源码的时候,想要修改其中被 final 修饰符修饰的字段的值,可行吗?
方案
使用Java反射,通过 Field#setAccessible(true)
将 private 修饰的字段变为 accessible;再将 final 修饰符去掉;最后再设置新值即可。当然,如果涉及到 Java 内联优化,则会失效。具体见示例代码:
package com.tianma.sample;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
public class ChangeStaticFinalFieldSample {
static void changeStaticFinal(Field field, Object newValue) throws Exception {
field.setAccessible(true); // 如果field为private,则需要使用该方法使其可被访问
Field modifersField = Field.class.getDeclaredField("modifiers");
modifersField.setAccessible(true);
// 把指定的field中的final修饰符去掉
modifersField.setInt(field, field.getModifiers() & ~Modifier.FINAL);
field.set(null, newValue); // 为指定field设置新值
}
public static void main(String[] args) throws Exception {
Sample.print();
Field canChangeField = Sample.class.getDeclaredField("CAN_CHANGE");
Field cannotChangeField = Sample.class.getDeclaredField("CANNOT_CHANGE");
changeStaticFinal(canChangeField, 2);
changeStaticFinal(cannotChangeField, 3);
Sample.print();
}
}
class Sample {
private static final int CAN_CHANGE = new Integer(1); // 未内联优化
private static final int CANNOT_CHANGE = 1; // 内联优化
public static void print() {
System.out.println("CAN_CHANGE = " + CAN_CHANGE);
System.out.println("CANNOT_CHANGE = " + CANNOT_CHANGE);
System.out.println("------------------------");
}
}
打印结果为:
CAN_CHANGE = 1
CANNOT_CHANGE = 1
------------------------
CAN_CHANGE = 2
CANNOT_CHANGE = 1
------------------------
通过以上输出结果可以看出, CAN_CHANGE
和 CANNOT_CHANGE
字段同属于 final 修饰符修饰的常量字段,但是由于 CANNOT_CHANGE
常量在 Java 编译过程中使用了内联优化,其值在编译阶段就被编译为常量值 1
,故而使用内联优化的 final 字段更改其值是无效的; 而 CAN_CHANGE
字段未被内联优化,故而能通过 Java 反射对其值进行修改。