首先来看一下jdk1.8中的String类的源码
我们可以看到String类是被final修饰的,说明String类是不能够被继承的。
另外,我们可以看到用于存储数据的是一个char[]数组,它也是被final修饰的,这说明它是不能修改指向的地址的,(但是可以被直接修改内部的字符变量),为了保证String对象不被改变,有以下措施:
(1)不对外暴露value数组中的值
(2)将String类设置为final,不可被继承,避免子类继承后改变value数组中的值。
那么为什么String类要被final修饰呢?
1 为了构建字符串常量池
创建字符串时,如果该字符串已经存在于池中,则将返回现有字符串的引用,而不是创建新对象。可以有多个String对象指向同一个内存地址。 但如果字符串是可变的,那么字符串常量池将不能实现,因为这样的话,如果变量改变了它的值,那么其它指向这个值的变量的值也会一起改变。
2 可以缓存hashCode
字符串的Hashcode在java中经常配合基于散列的集合一起正常运行,这样的散列集合包括HashSet、HashMap以及HashTable。不可变的特性保证了hashcode永远是相同的。不用每次使用hashcode就需要计算hashcode。这样更有效率。因为当向集合中插入对象时,是通过hashcode判别在集合中是否已经存在该对象了(不是通过equals方法逐个比较,效率低)。
3 保证线程安全
由于String类是不可变的,因此是多线程安全的,同一个字符串实例可以被多个线程共享。这样便不用因为线程安全问题而使用同步。字符串自己便是线程安全的。
4 保证数据安全
如果字符串是可变的,那么会引起很严重的安全问题。譬如,数据库的用户名、密码都是以字符串的形式传入来获得数据库的连接,或者在socket编程中,主机名和端口都是以字符串的形式传入。因为字符串是不可变的,所以它的值是不可改变的,否则黑客们可以钻到空子,改变字符串指向的对象的值,造成安全漏洞。