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

众所周知:

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());

    }

看一下效果:

 

 

好了,该篇就到此。 

posted on 2022-11-08 07:34  小目标青年  阅读(23)  评论(0编辑  收藏  举报