众所周知:

Stirng是个不可变的类,因为使用了final来修饰(真的只是这个一个final的功劳吗? ),如:

Java  String 为什么不可变? 真的吗?_数组

又有一个众所周知:

就是 String的本质是一个char[] 数组。

所以为了确保String真的不可变,那么本质肯定不能变,于是乎这个char[]数组,如:

Java  String 为什么不可变? 真的吗?_不可变_02

而且这个成员变量是没有提供set和get方法。

 

 

看到这里,其实可以知道的就是,如果这个value数组的元素被改变了,那么String就是被改变了。

所以在String的源码里,诸多方法里面都没有涉及到直接去修改value[]的元素。

 

再度聚焦:

Java  String 为什么不可变? 真的吗?_成员变量_03

这个构造函数,一眼能看明白意思,就是通过传递一个char 数组,进行构造出一个新的String。

但是又细眼一看? 


Java  String 为什么不可变? 真的吗?_成员变量_04

Arrays的copyOf方法去实现深拷贝:

Java  String 为什么不可变? 真的吗?_成员变量_05

 这样做的原因大家都知道,就是重新开辟一波新的空间,这样防止 在后续修改传入的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());

}

看一下效果:

Java  String 为什么不可变? 真的吗?_String_06

 

 

好了,该篇就到此。