同样的代码,在jdk8能正常修改String对象,而在jdk21无法修改成功。
//jdk8
@Test
public void testReflect () {
String abc = "123";
String abcCopy = "123";
System.out.println("abc==" + abc + " , abcCopy==" + abcCopy);
Field value = ReflectionUtils.findField(String.class, "value");
ReflectionUtils.makeAccessible(value);
char[] field = (char[])ReflectionUtils.getField(value, abc);
for (int i=0; i<field.length; i++) {
field[i] += 1;
}
System.out.println("finally, abc==" + abc + " , abcCopy==" + abcCopy);
}
输出结果:
abc==123 , abcCopy==123
finally, abc==234 , abcCopy==234
//jdk21,与jdk8的小区别是存储内容的使用了byte[]数组了,而不是char[]数组
@Test
void contextLoads() {
String abc = "123";
String abcCopy = "123";
System.out.println("abc==" + abc + " , abcCopy==" + abcCopy);
Field value = ReflectionUtils.findField(String.class, "value");
ReflectionUtils.makeAccessible(value);
byte[] field = (byte[])ReflectionUtils.getField(value, abc);
for (int i=0; i<field.length; i++) {
field[i] += 1;
}
System.out.println("finally, abc==" + abc + " , abcCopy==" + abcCopy);
}
输出结果
abc==123 , abcCopy==123
java.lang.reflect.InaccessibleObjectException: Unable to make field private final byte[] java.lang.String.value accessible: module java.base does not "opens java.lang" to unnamed module @27c86f2d
可能是由于特定的JVM实现、内存布局或其他不可预测的行为所导致的。在标准的Java实现中,String
对象是不可变的,并且通过反射修改其内部状态是不应该影响外部可见性的。有些情况下,由于Java内存模型和JVM实现的具体细节,可能会观察到一些看似不合逻辑的行为。