Vamos a empezar con un caso de prueba sencilla:Modificación de campos finales en Java
import java.lang.reflect.Field;
public class Test {
private final int primitiveInt = 42;
private final Integer wrappedInt = 42;
private final String stringValue = "42";
public int getPrimitiveInt() { return this.primitiveInt; }
public int getWrappedInt() { return this.wrappedInt; }
public String getStringValue() { return this.stringValue; }
public void changeField(String name, Object value) throws IllegalAccessException, NoSuchFieldException {
Field field = Test.class.getDeclaredField(name);
field.setAccessible(true);
field.set(this, value);
System.out.println("reflection: " + name + " = " + field.get(this));
}
public static void main(String[] args) throws IllegalAccessException, NoSuchFieldException {
Test test = new Test();
test.changeField("primitiveInt", 84);
System.out.println("direct: primitiveInt = " + test.getPrimitiveInt());
test.changeField("wrappedInt", 84);
System.out.println("direct: wrappedInt = " + test.getWrappedInt());
test.changeField("stringValue", "84");
System.out.println("direct: stringValue = " + test.getStringValue());
}
}
Cualquiera cuidado para adivinar lo que va a ser impreso como salida (que se muestra en la parte inferior, para no estropear la sorpresa de inmediato).
Las preguntas son:
- ¿Por qué número entero primitiva y envuelto comportan de manera diferente?
- ¿Por qué el acceso directo reflectante arroja resultados diferentes?
- El que más me molesta, ¿por qué String se comporta como primitivo
int
y no comoInteger
?
resultados (Java 1.5):
reflection: primitiveInt = 84
direct: primitiveInt = 42
reflection: wrappedInt = 84
direct: wrappedInt = 84
reflection: stringValue = 84
direct: stringValue = 42
+1: agradable leer su tema. –
esto puede estar estrechamente relacionado con el compilador. – andy