问题
起初反射修改String的final变量,按理应该能够修改的,但是发现修改不了?但是如果把String赋值形式改为new String("xxx");,则可以,又是为什么?
为什么反射修改 final String str = "" 变量修改不了,而 final String str = new String("xxx") 形式可以修改?
为什么反射修改非static的final变量不消去final不会报错,而static&final变量不消去final修饰符会报错?
结论
反射修改final变量和是否static无关
反射修改字段值,需要获取Field,底层是FieldAccessor接口,不同类型Field不同FieldAccessor实现,所以会导致非static的final变量不消去final不会报错,而static&final变量不消去final修饰符会报错
具体值修改是通过unsafe.putObjectVolatile(var1, this.fieldOffset, var2);,所以修改类型底层数据结构,存储结构有关。因为final String str = "xxx" 是存储在方法区中的常量池中,是常量,而final String str = new String("xxx");也是存储在方法区中,但是是引用类型,所以引用类型的话就可以直接修改引用,而常量则不可以。
注意:数组是引用类型。
问题解决方式
问题很奇怪,刚开始想了很久没思路,觉得debug一下
public void set(Object obj, Object value)
throws IllegalArgumentException, IllegalAccessException {
// 检查
if (!override) {
if (!Reflection.quickCheckMemberAccess(clazz, modifiers)){
Class> caller = Reflection.getCallerClass();
checkAccess(caller, clazz, obj, modifiers);
}
}
// 获取FieldAccessor,并修改值
getFieldAccessor(obj).set(obj, value);
}
修改Field接口:FieldAccessor很关键
public FieldAccessor newFieldAccessor(Field var1, boolea