您的位置:首页 > 其它

深入理解String为什么是不可变的类

2016-05-13 18:10 369 查看




如上图所示

String s1="abcd" ;    等价于  String s1=new String("abcd");

两个变量都指向同一个堆内存



String s="abcd";

s=s.concat("ef");

是如下的流程:





当一个字符串在堆中被分配内容时,它就是不可变的,任何String的方法都无法改变字符串本身,但它可以返回一个新的字符串对象。

如果需要可以修改的字符串对象,可以使用StringBuffer和StringBuilder,StringBuffer是线程安全的,由于StringBuilder不需要进行同步操作,StringBuilder是比较快速的。

综上所述:

1)字符串池(String pool)的需求 在Java中,当初始化一个字符串变量时,如果字符串已经存在,就不会创建一个新的字符串变量,而是返回存在字符串的引用。 例如: String s1="abcd"; String s2="abcd";
这两行代码在堆中只会创建一个字符串对象。如果字符串是可变的,改变另一个字符串变量,就会使另一个字符串变量指向错误的值。

 2)缓存字符串hashcode码的需要 字符串的hashcode是经常被使用的,字符串的不变性确保了hashcode的值一直是一样的,在需要hashcode时,就不需要每次都计算,这样会很高效。

 3)出于安全性考虑 字符串经常作为网络连接、数据库连接等参数,不可变就可以保证连接的安全性。 

JAVA中的参数传递全是以值传递的



 总的来说,简单概括Java中进行方法调用的时候传递参数时,遵循值传递的原则: 
        1)基本数据类型,传递的是数据的拷贝 。“值传递”,在这种方式下,被调用对象对新数据的改变不影响源数据的取值,如基本类型的传递

//例1 
void method1(){ 
int x=0; 
this.change(x); 
System.out.println(x); 


void int change(int i){ 
i=1; 

        结果:x=0; 
        2)引用数据类型,传递的是传递的引用地址的拷贝,而不是该对象本身 。被调用对象新数据的改变影响源数据内容,因为新数据和源数据的引用虽然不同但却指向同一数据对象,如类型对象的传递。

void
method1(){ 
StringBuffer x=new StringBuffer("Hello"); 
this.change(x); 
System.out.println(x); 


void int change(StringBuffer i){ 
i.append(" world!"); 

        结果:x=Hello world; 

String是一个比较特殊的数据类型,在Java中,String是一个类,但是他的传值却是采用“值传递”的方式。 
首先String不属于8种基本数据类型,String是一个对象。


 

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: