之前在一个项目中,遇到一个问题。有些final类型的配置属性需要热更重新赋值,结果有一部分属性总是赋值不生效。
问题重现
测试对象:
测试代码:
修改final属性的方法
输出结果:
反射修改结果:
我们可以发现 反射修改的值是成功了,但是最后输出结果却不对。
我们反编译UserInfo.class 看一下toString()
问题分析
我们发现 name,sex属性在编译时刻就把值直接放到了引用它的地方。这就是 Java 编译器对 final 属性做了内联优化,即编译时把该 final 属性的值直接放到了引用它的地方。即使是反射修改了该属性,但是不能修改最终引用地方的值,所以输出结果不对。
问题解决
我们要想有效的给final属性重新赋值,那我们就需要规避掉内联优化。
Java会对基本类型byte, char, short, int, long, float, double, boolean; 再加上 Literal String 类型(直接双引号字符串) 做final类型属性内联优化。
我们应该如何避免内联优化?
a.构造方法赋值。b.使用基本类型相应的包装类型,直接字符串对象使用new String()替代(这一条会消耗更多的内存、及CPU 不建议使用)。c.使用赋值方法赋值(参考UserInfo. defautlValue(T value))。
注意:内联优化跟属性是否是static 无关
希望本文的内容对大家能带来一定的帮助,如果有疑问大家可以留言交流。
示例代码地址:https://github.com/xiaozhumt/example.git