本文主要来介绍下java中的不可变对象,以及java中的String类的不可变性,那么为什么Java中String类是不可变对象?让我们一起来分析下
原因一:
最流行的Java面试题之一就是:什么是不可变对象(immutable object),不可变对象有什么好处,在什么情况下用,或者更具体一点,Java的String类为什么要设置成immutable类型?不可变对象,顾名思义就是创建后无法改变的对象,典型的例子就是String类。
String s="XIAONAN";s.toLowerCase();s.toLowerCase()并没有改变'XIAONAN'的值,而是创建新的String类“xiaonan”,然后将新的实例指向变量s。
相对于可变对象,不可变对象有很多优势:
(1)不可变对象可以提高String Pool的效率和安全性。如果你知道一个对象是不可变的,那么需要拷贝这个对象的内容时,就不用复制它的本身而是复制它的地址,复制地址(通常一个指针的大小)需要很小的内存,效率也很高。对于同时引用“XIAONAN”的其他变量也不会造成影响。
(2)不可变对象对于多线程是安全的,因为在多线程同时进行的情况下,一个可变对象的值很可能被其他进程改变,这样会造成不可预期的结果,而使用不可变对象就可以避免这样的情况。
当然也有其他方面的原因,但是Java把String设置成immutable变量最大的原因应该是效率与安全性。
原因二:
这是一个老生常谈的话题(This is an old but still popular question),在Java中将String设置成不可变对象是综合考虑到各种因素的结果,想要理解这个问题,需要综合“内存”、“同步”、“数据结构”、以及“安全”等方面来考虑,在下文中,我将为各种原因做一个总结。
1.字符串常量池的需要
字符串常量池(String Pool,String intern pool,String 保留池)是java堆内存中一个重要的存储区域,当创建一个String对象时,假如此字符串值已经存在于常量池中,则不会创建一个新的对象,而是引用已经存在的对象。
如下面的代码所示,将会在堆内存中只创建一个实际的String对象。
String s1=“abcd”;
String s2=“abcd”;
若变换一种方式,s1和s2还会指向同一个实际的String对象吗?如下所示
String s1=“ab”+“cd”;
String s2=“abc“+”d“;
假若字符串对象允许改变,那么将会导致各种逻辑错误,比如改变一个对象会影响到另一个独立对象,严格来说,这种常量池的思想,是一种优化手段。
也许这个问题违反新生的直觉,但考虑到现代编译器会考虑到常规的优化,所以他们都会指向常量中的同一个对象,或者,你可以用jd-gui之类的工具查看一下编译后的class文件。
2.允许String对象缓存HashCode
Java中String对象的哈希码被频繁地使用, 比如在hashMap 等容器中。
字符串不变性保证了hash码的唯一性,因此可以放心地进行缓存.这也是一种性能优化手段,意味着不必每次都去计算新的哈希码. 在String类的定义中有如下代码:
private int hash;//用来缓存HashCode
3.安全性
String被许多的Java类(库)用来当做参数,例如 网络连接地址URL,文件路径path,还有反射机制所需要的String参数等, 假若String不是固定不变的,将会引起各种安全隐患。