1. String为什么不可变
String的源码
public final class String {
// 用字符数组来存储值
private final char value[];
}
虽然String是final类型的类,且value也是final类型的数组,但这不是String不可变的根本原因,String不可变是因为value是private,且并没有提供对外的get和set
被final修饰的类、变量、属性有以下特点
- 被final修饰的类不能被继承
- 被final修饰的方法不能被重写
- 被final修饰的变量不能被修改(java只有值传递,但对于基本类型来说这个值就是变量自身的值,对引用类型来说,这个值指的值对象的引用地址。)
2. 怎样可以改变String的值
利用反射可以改变String中的值。虽然业务上没有几乎没有人这么做,但是确实可以实现。
对于引用类型,会在堆中创建实例,然后对象的具体指只在栈中进行保存。堆中保存的值栈中对应的引用地址。
String中的value是引用类型,所以可以做到在不改变堆中引用的情况下,改变栈中具体的值即可。
代码实例
public class StringModifyTest {
public static void main(String[] args) throws Exception{
// 1. mock一个String
String str = "Melodfy";
// 2. 通过反射过去对象
// 反射获取对象的三种方法
// 1. Class.forName("类全限定名")
// 2. 类名.class()
// 3. 实例.getClass()
Class<? extends String> strCls = str.getClass();
// 3. 通过反射获取指定参数,得到一个属性对象
Field field = strCls.getDeclaredField("value");
// 私有属性要设置可访问,否则访问操作时异常
filed.setAccessible(true);
// 4. 根据属性对象获取参数的具体指
char[] result = (char[])field.get(str);
// 5. 验证输出结果
Systemg,out.println("变更前结果:");
for(char c : result) {
System.out.print(c);
}
// 6. 操作变更,然后输出验证
System.out.println("变更后结果:");
char[2] = 'w';
for(char c : result) {
System.out.print(c);
}
}
}
控制台输出
变更前结果:
Melody
变更后结果:
Mwlody
3. 总结
- String不可变是因为属性私有,且没有对外提供编辑入口
- 可以通过反射修改String中的字符(万能的反射)