Java String 为什么不可变? 真的吗?
原创
©著作权归作者所有:来自51CTO博客作者小目标青年来也的原创作品,请联系作者获取转载授权,否则将追究法律责任
众所周知:
Stirng是个不可变的类,因为使用了final来修饰(真的只是这个一个final的功劳吗? ),如:
又有一个众所周知:
就是 String的本质是一个char[] 数组。
所以为了确保String真的不可变,那么本质肯定不能变,于是乎这个char[]数组,如:
而且这个成员变量是没有提供set和get方法。
看到这里,其实可以知道的就是,如果这个value数组的元素被改变了,那么String就是被改变了。
所以在String的源码里,诸多方法里面都没有涉及到直接去修改value[]的元素。
再度聚焦:
这个构造函数,一眼能看明白意思,就是通过传递一个char 数组,进行构造出一个新的String。
但是又细眼一看?
Arrays的copyOf方法去实现深拷贝:
这样做的原因大家都知道,就是重新开辟一波新的空间,这样防止 在后续修改传入的char value[] 里的元素,导致String也跟着被修改(如果写成 this.value=value)
霸王硬上弓:
那么咱们就是要修改String,怎么办?
那肯定就是修改它的本质 char[] value的元素了。
咱们通过反射去修改String的成员变量,也就是这个本质 char数组,一起来看看:
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
String testStr = "JCccc";
System.out.println("一开始的testStr值为 : " +testStr);
System.out.println(testStr.hashCode());
//反射机制,获取获取String里面的的value字段
Field valueFieldOfString = String.class.getDeclaredField("value");
//设置value属性的访问权限为true
valueFieldOfString.setAccessible(true);
//获取s对象上的value属性的值
char[] value = (char[]) valueFieldOfString.get(testStr);
//改变value数组中的元素
value[1] = 'A';
value[2] = 'a';
System.out.println("被操作之后的testStr为 : " +testStr);
System.out.println(testStr.hashCode());
}
看一下效果:
好了,该篇就到此。