通过Java反射修改字符串内容:
public static void main(String[] args) throws Exception {
String str = new String("abc");
Field value = str.getClass().getDeclaredField("value");
// 默认false,设置为true为打破封装,访问私有属性
value.setAccessible(true);
value.set(str,"abcd");
System.out.println(str);
}
执行报错:
Exception in thread "main" java.lang.IllegalArgumentException: Can not set final [B field java.lang.String.value to java.lang.String
at java.base/jdk.internal.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:167)
at java.base/jdk.internal.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:171)
at java.base/jdk.internal.reflect.UnsafeQualifiedObjectFieldAccessorImpl.set(UnsafeQualifiedObjectFieldAccessorImpl.java:83)
at java.base/java.lang.reflect.Field.set(Field.java:780)
因为JVM在编译时期, 就把final类型的String进行了优化, 在编译时期就会把String处理成常量。所以无法修改类似String str = "123456"的值。
官方文档也说明指定对象是声明底层字段(或者其子类或实现者)的类或接口的实例才能修改String的值。
public class People {
String str = "123";
}
public static void main(String[] args) throws Exception {
People p = new People();
System.err.println(p.str);// 123
Field field = People.class.getDeclaredField("str");
field.setAccessible(true);
field.set(p, "12345");
System.err.println(p.str);// 12345 修改String值成功
}