我正在探索java反射API,我遇到了以下代码片段
public class Main {
public static void main(String[] args) throws IllegalAccessException, NoSuchFieldException{
Field value=Integer.class.getDeclaredField("value");
value.setAccessible(true);
value.set(42, 43);
System.out.printf("six times seven %d%n",6*7);
System.out.printf("six times seven %d%n",42);
System.out.println(42);
}
}
输出:
six times seven 43
six times seven 43
42
我阅读了set方法的文档,该方法声明它为给定对象设置字段的值。但我无法理解代码的输出,因为它应该在所有情况下打印42。
任何人都可以深入了解代码中发生的事情吗?
答案 0 :(得分:4)
System.out.println(42);
正在调用println(int)
而不是println(Object)
。拳击永远不会发生。这使它更快,并且在1.5之前也工作。
在其他情况下,您正在通过Integer.valueOf(int)
进行拳击。此方法被定义为始终为-128和127之间的值返回完全相同的Integer
对象(对于其他值可能具有或可能不具有相同的行为)。因此,无论您在程序中装箱42,您都将获得相同的对象,当您在此对象中设置value
的值时,无论读取哪个参考,它都会发生变化。
如果你明确地将拳击放入代码中,它将如下所示:
value.set(Integer.valueOf(42), 43);
System.out.printf("six times seven %d%n",Integer.valueOf(6*7));
System.out.printf("six times seven %d%n",Integer.valueOf(42));
System.out.println(42);
正如我们所知Integer.valueOf(
为42返回完全相同的对象,代码实际上是:
Integer obj42 = Integer.valueOf(42);
value.set(Integer.valueOf(obj42, 43);
System.out.printf("six times seven %d%n", obj42);
System.out.printf("six times seven %d%n", obj42);
System.out.println(42);