Java中的變量類型基本分為兩種:基本類型和引用類型。
- 基本類型為全小寫字母,如:int, long, short, char等;
- 引用類型為首字母大,如:String, Integer, User 等。
其中引用類型中又有一些是“不可變引用類型”,如:String, Integer等。
對于基本類型作為方法的參數(shù)傳遞的是值本身,沒什么問題。
對于引用類型作為方法的參數(shù)傳遞的是地址,看下面的例子。
Demo1
String s = "abc";
changeValue(s);
System.out.println(s); //abc
public static void changeValue(String str){
str+="cde";
}
在調(diào)用changeValue方法的時候,會將變量s指向的字符串“abc”的地址3a傳給局部變量str,這時str也指向字符串“abc”。而在Java中String是不可改變的,若改變String類型變量的值,只能是在字符串常量池中新創(chuàng)建一個字符串“abccde”。而這時,str通過地址3b指向了字符串常量“abccde”。而s依舊通過地址3a指向字符串常量“abc”。因此調(diào)用端輸出s的值為“abc”(未改變)。
Demo2
StringBuffer sb = new StringBuffer("abc");
changeValue(sb);
System.out.println(sb); //abccde
public static void changeValue(StringBuffer str){
str.append("cde");
}
在調(diào)用changeValue的時候,同樣會吧sb指向的對象的堆內(nèi)存地址3a傳給局部變量str,這時str也指向該塊地址。然后調(diào)用str的append方法時,在該堆內(nèi)存區(qū)域中追加了字符串“cde”,3a處的對象本身改變了。因此在調(diào)用端輸出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;
}
在調(diào)用swap方法的時候,會將sb1指向的對象的內(nèi)存地址3a,sb2指向的內(nèi)存地址3b分別傳給局部變量s1,s2。接下來,通過中間局部變量sb將s1和s2的值交換了,交換后s1指向內(nèi)存地址3b,s2指向內(nèi)存地址3a。而從始至終sb1和sb2所指向的內(nèi)存地址并未改變,sb1—>3a,sb2—>3b。所以,在調(diào)用端輸出的時候,sb1輸出“111”,sb2輸出“222”。
結(jié)論
有人說Java中參數(shù)的傳遞都是值傳遞,不存在引用傳遞。也有人不這么認為。然而,只要理解了參數(shù)傳遞的本質(zhì)就行了。
首先,基本類型的參數(shù),傳遞的肯定是值本身,因為基本類型的變量存在JVM的棧內(nèi)存中。不存在堆內(nèi)存地址的概念;
其次,引用類型的參數(shù)(不管是可變引用類型還是不可變引用類型)傳遞的都是該變量所指向的堆內(nèi)存的地址。而在調(diào)用方法的時候,會將這個地址的值賦給方法的局部變量,使得這個局部變量也指向相應的堆內(nèi)存。
只不過,如果是不可變引用類型的參數(shù),當對局部變量賦新值時,本質(zhì)上是開辟了新的內(nèi)存區(qū)域放入新值,使局部變量指向新的內(nèi)存區(qū)域,因此不會改變調(diào)用端變量的值。
而如果是可變引用類型的參數(shù),當改變對象的內(nèi)容時,改變是這個對象所在內(nèi)存中的值,因此,在調(diào)用端讀取的時候是改變后的值。