Java中的变量类型基本分为两种:基本类型和引用类型。
- 基本类型为全小写字母,如:int, long, short, char等;
- 引用类型为首字母大,如:String, Integer, User 等。
其中引用类型中又有一些是“不可变引用类型”,如:String, Integer等。
对于基本类型作为方法的参数传递的是值本身,没什么问题。
对于引用类型作为方法的参数传递的是地址,看下面的例子。
Demo1
String s = "abc";
changeValue(s);
System.out.println(s); //abc
public static void changeValue(String str){
str+="cde";
}
在调用changeValue方法的时候,会将变量s指向的字符串“abc”的地址3a传给局部变量str,这时str也指向字符串“abc”。而在Java中String是不可改变的,若改变String类型变量的值,只能是在字符串常量池中新创建一个字符串“abccde”。而这时,str通过地址3b指向了字符串常量“abccde”。而s依旧通过地址3a指向字符串常量“abc”。因此调用端输出s的值为“abc”(未改变)。
Demo2
StringBuffer sb = new StringBuffer("abc");
changeValue(sb);
System.out.println(sb); //abccde
public static void changeValue(StringBuffer str){
str.append("cde");
}
在调用changeValue的时候,同样会吧sb指向的对象的堆内存地址3a传给局部变量str,这时str也指向该块地址。然后调用str的append方法时,在该堆内存区域中追加了字符串“cde”,3a处的对象本身改变了。因此在调用端输出sb的时候,表示输出3a处的对象,因此,输出的是改变后的值。
Demo3
StringBuffer sb1 = new StringBuffer("111");
StringBuffer sb2 = new StringBuffer("222");
swap(sb1, sb2);
System.out.println("sb1="+sb1);
System.out.println("sb2="+sb2);
public static void swap(StringBuffer s1, StringBuffer s2){
StringBuffer sb = s1;
s1=s2;
s2=sb;
}
在调用swap方法的时候,会将sb1指向的对象的内存地址3a,sb2指向的内存地址3b分别传给局部变量s1,s2。接下来,通过中间局部变量sb将s1和s2的值交换了,交换后s1指向内存地址3b,s2指向内存地址3a。而从始至终sb1和sb2所指向的内存地址并未改变,sb1—>3a,sb2—>3b。所以,在调用端输出的时候,sb1输出“111”,sb2输出“222”。
结论
有人说Java中参数的传递都是值传递,不存在引用传递。也有人不这么认为。然而,只要理解了参数传递的本质就行了。
首先,基本类型的参数,传递的肯定是值本身,因为基本类型的变量存在JVM的栈内存中。不存在堆内存地址的概念;
其次,引用类型的参数(不管是可变引用类型还是不可变引用类型)传递的都是该变量所指向的堆内存的地址。而在调用方法的时候,会将这个地址的值赋给方法的局部变量,使得这个局部变量也指向相应的堆内存。
只不过,如果是不可变引用类型的参数,当对局部变量赋新值时,本质上是开辟了新的内存区域放入新值,使局部变量指向新的内存区域,因此不会改变调用端变量的值。
而如果是可变引用类型的参数,当改变对象的内容时,改变是这个对象所在内存中的值,因此,在调用端读取的时候是改变后的值。