Java反射修改final修饰的属性值

版权声明:本文源自tianma,转载请务必注明出处: http://www.jianshu.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_CHANGECANNOT_CHANGE 字段同属于 final 修饰符修饰的常量字段,但是由于 CANNOT_CHANGE 常量在 Java 编译过程中使用了内联优化,其值在编译阶段就被编译为常量值 1,故而使用内联优化的 final 字段更改其值是无效的; 而 CAN_CHANGE 字段未被内联优化,故而能通过 Java 反射对其值进行修改。

参考

Change private static final field using Java reflection

你可能感兴趣的:(Java反射修改final修饰的属性值)