Cominciamo con un semplice caso di test:Modifica dei campi finali in 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());
}
}
Chiunque cura di intuire ciò che verrà stampato come uscita (mostrato in basso da non rovinare subito la sorpresa).
Le domande sono:
- Perché primitiva e avvolto intero si comportano in modo diverso?
- Perché l'accesso diretto riflessivo vs restituisce risultati diversi?
- Quello che mi affligge di più - perché String si comporta come il primitivo
int
e non comeInteger
?
Risultati (Java 1.5):
reflection: primitiveInt = 84
direct: primitiveInt = 42
reflection: wrappedInt = 84
direct: wrappedInt = 84
reflection: stringValue = 84
direct: stringValue = 42
+1: Piacere di leggere l'argomento. –
questo potrebbe essere strettamente correlato al compilatore. – andy