首先,java中有这两种类型数据。基本数据类型
整型:byte,short,int,long浮点型:float,double字符型:char布尔型:boolean引用数据类型
数组对象(类、接口)
一般情况下,基本数据类型传递时是值传递,引用数据类型是引用传递。
而String有些不一样,它是字符串对象,既然是对象,传递应该是引用传递,但是实际却不一样。
直接看例子。
public static void main(String[] args){
String s1 = "hello";
String s2 = "world"; System.out.println(s1+"---"+s2); //hello---world
change(s1,s2);
System.out.println(s1+"---"+s2); //hello---world
StringBuffer sb1 = new StringBuffer("hello");
StringBuffer sb2 = new StringBuffer("world"); System.out.println(sb1+"---"+sb2);//hello---world
change(sb1,sb2);
System.out.println(sb1+"---"+sb2);//***hello---worldworld
} private static void change(StringBuffer sb1,StringBuffer sb2){
sb1 = sb2;
sb2.appned(sb1);
}
public static void change(String s1,String s2){
s1 = s2;
s2 = s1+s2;
}
可以看出,String 的本质却是值传递,而StringBuffer是引用传递。
再看看String内部代码。
(借鉴的某图)
可以看出,其内部存储值得变量用了final修饰,所以也是常说的String的值不可变。
但是,测试一下,可以发现。
String str = "ab";
String str = "abcd";
而这两行代码却不会报错,可以说是str的值被改变了。
但实际上却是,新建了一个名为"abcd"的字符串放在了常量池中,让str指向了它,而常量池中还有个"ab",这个"ab"是str原本引用的值,它并没有任何的改变。
并不是说str的值被改变了,而是它引用的地址被改变了,String是个对象类型,也有这种原因。
所以最开始的例子,说它是对象,对象是引用传递,这并没有错,可实际上它是值传递,为什么?
在参数传递中,Stirng 实参地址 传给了 形参 ,形参有了地址,在change方法中,它在做运算的时候(s2 = s1+s2) 却新建了一个字符串"helloworld" 放在常量池中,再加上之前的两个(s1,s2),一共有三个地址。change方法中的s2指向"helloworld",但主方法中的s1、s2依然指向的原来的地址("hello"、"world"),其中的内容没有任何的改变,所以说是值传递。
而StringBuffer,实参地址 传给了 形参 ,形参有了地址,在change方法中,它在做运算的时候,没有新建字符串,在常量池中还是只有两个地址,StringBuffer是直接就在s2的地址上的值进行修改。
为什么String会在做运算的时候会新建字符串呢?
这是因为String类 在做运算的时候,对操作符进行了重载。