java函數參數傳遞的到底是值還是引用對確實容易讓人迷糊。而很多時候因為對這個問題的模糊甚至造成一些錯誤。最常見的說法是基本類型傳的是值,對象傳的引用。對於基本類型,大家都達成共識,沒有什麼可以爭論的。但對於對象傳遞,還有一說是也是值傳遞。首先來看一個例子。一個簡單類person,只有一個屬性name,和一個人事系統。如下所示:
public class Person{ private String name; public Persion(String name){ this.name = name; } public String getName(){ return this.name; } public void setName(String name){ this.name = name; } }
public class PersonnalSystem{ public static void changeName1(Persion person){ person.setName("Bob"); System.out.println("person name in changeName1: " person.getName());// Bob,被修改了 } public static void changeName2(Persion person){ Person anotherPerson = new Persion("Tom"); person = anotherPerson; System.out.println("person name in changeName2: " person.getName());// Tom被修改了 } public static void main(String[] agrs){ Person person = new Persion("Lily"); System.out.println("before change name person's name is"+person.getName);//Lily changeName2(person); System.out.println("after first change name person's name is"+person.getName);//被修改Bob changeName2(person); System.out.println("after second change name person's name is"+person.getName);//仍然是Bob,沒有修改 } }
運行這個程序可以看到,changeName1修改了person的名字,確實是傳遞引用。但是當把person傳遞到changeName2中雖然在方法內修改成功,但main函數中仍然是Bob沒有修改。這似乎又是傳值了。這是因為,對象傳遞過程中傳遞的是對象的地址,而非對象的引用,也不是對象的副本。假設剛開始person的地址為60,changeName1接收到的參數是地址60,所以修改是在地址60的地方進行的,因此對象改變。而在changeName2中新建anotherPerson的地址也許是65,但絕非是60.當把anotherPerson賦值給person時,只是將person所代碼的60改變成65,後續的修改操作都是在65處進行的,因此對main(60)中的person無影響。
所以總之來說我認為java參數傳遞過程是值傳遞。
這種方式對於String,Integer, Double等immutable類型時,也是符合的。因為這些對象不可變,因此每次傳遞的都是值。當然如果非要認為對象是引用傳遞也未嘗不可,但是首先要區別String這類不可變對象,其次還要主義changeName2這種情況下千別希望原來對象被改變。