为什么将String设计为不可变

谈到String为不可变对象,有一个容易混淆的误区,例如:

为什么将String设计为不可变

这里a只是一个对象的引用,最明显的比如Class cls;这只是创建了Class对象的引用,Class cls = new Class();才是创建Class对象。所以这里其实是创建了两个字符串对象。同样的误区还有对字符串的一些操作方法后,substring(),replace()等,其实这都是重新创建了字符串对象,与字符串对象不可变并不悖论。从replace()的源码中可以看出来。

为什么将String设计为不可变

关于为什么String对象不可变,可以看看它的设计源码。

为什么将String设计为不可变

可以看出String对象本质上就是一个字符数组,这也和数组就是对象相符,也可以解释了String.charAt()等方法的原理。value[]就是这个字符数组,offset是在数组中的起始位置,count是字符数组的长度即字符串的长度,而hash是这个String对象的hash值缓存。value[],offset和count这三个变量都是private而且没有setter方法,这也是String对象不可变的本质原因。

那将String对象设计为不可变有什么好处呢?首先就是刚提到的源码中hash变量。String对象如果改变就必须重新计算hash值,这是比较麻烦的,这是性能方面。安全方面。String对象不可变自然是多线程并发安全的。此外,程序中类加载,数据库连接等资源的引入经常要用到字符串,将字符串设计为不可变也保证了这方面的安全。

那么,String真的不可变吗?其实可以通过反射去修改私有变量!不过我们一般不会去考虑。