問: 什么是值傳遞妖枚?什么是引用傳遞?為什么說 Java 中只有值傳遞苍在?
一绝页、值傳遞與引用傳遞
實參與形參:
- 實際參數(shù):在調(diào)用有參函數(shù)時,主調(diào)函數(shù)和被調(diào)函數(shù)之間有數(shù)據(jù)傳遞關(guān)系寂恬。在主調(diào)函數(shù)中調(diào)用被調(diào)函數(shù)時续誉,函數(shù)名后面括號中的參數(shù)稱為“實際參數(shù)”。
- 形式參數(shù):在定義函數(shù)名和函數(shù)體時使用的參數(shù)初肉,目的是用來接收調(diào)用此函數(shù)時傳入的參數(shù)酷鸦。
值傳遞與引用傳遞:
- 值傳遞:是指在調(diào)用函數(shù)時,將實際參數(shù)復(fù)制一份,通過形參傳遞到函數(shù)中臼隔。這時形參接收到的內(nèi)容是實參的一個副本嘹裂,這樣在函數(shù)中如果對形參進行修改,不會影響到實際參數(shù)躬翁。
- 引用傳遞:是指在調(diào)用函數(shù)時焦蘑,將實際參數(shù)的地址傳遞到函數(shù)中,那么在函數(shù)中對參數(shù)進行的修改盒发,將會影響到實際參數(shù)例嘱。
重點的區(qū)別在于:
方面 | 值傳遞 | 引用傳遞 |
---|---|---|
根本區(qū)別: | 會創(chuàng)建副本(copy) | 不會創(chuàng)建副本 |
所以: | 函數(shù)中無法改變原始對象 | 函數(shù)中可以改變原始對象 |
二、Java 中的值傳遞
這里需要清楚 JVM 內(nèi)存的劃分及職能宁舰,即:
- 虛擬機棧
- 堆
- 程序計數(shù)器
- 方法區(qū)
- 本地方法棧
已知拼卵,JVM 會給每個線程分配一個 Java 棧,當(dāng)執(zhí)行一個方法時蛮艰,JVM 會往該棧中壓入一個棧幀腋腮。
1. 對于基本數(shù)據(jù)類型
在方法中,基本數(shù)據(jù)類型的形參及變量(稱為局部變量)的名字和值都存儲于棧中壤蚜。在該方法被調(diào)用時即寡,會為形參在棧幀中開辟一塊內(nèi)存,將實參值復(fù)制給形參袜刷,然后形參和實參之間便沒有了關(guān)聯(lián)聪富。
即,此后在方法中對形參的操作不會影響到原來的實參著蟹。 例:
public static void main(String[] args) {
int a = 21;
intTest(a);
System.out.println("方法執(zhí)行后的a: "+a);
}
public static void intTest(int var){
// 將實參值復(fù)制給形參
System.out.println("傳入的形參var:"+var);
// 在方法中對形參的操作不會影響到原來的實參
var = 23;
System.out.println("方法內(nèi)重新賦值后的形參var:"+var);
}
運行結(jié)果:
傳入的形參var:21
方法內(nèi)重新賦值后的形參var:23
方法執(zhí)行后的a: 21
2. 對于引用數(shù)據(jù)類型(對象類型)
Java 里面的變量墩蔓,要么是基本數(shù)據(jù)類型,要么是指向?qū)ο髮嵗囊妙愋拖舳梗^對不會是一個對象奸披。
對于引用數(shù)據(jù)類型的對象,變量的名和值均存儲在棧中涮雷,變量值存儲的是對象的地址阵面,對象的實際內(nèi)容存儲于堆(堆區(qū)是共享的)中。在該方法被調(diào)用時洪鸭,會為形參在棧幀中開辟一塊內(nèi)存膜钓,將實參值復(fù)制給形參,即把實參指向的對象的地址復(fù)制給了形參:
1.此后在方法內(nèi)部卿嘲,如果沒有改變形參值(對象的地址)颂斜,那么可以通過該形參操作原實參指向的堆中的對象。這樣雖然原對象的屬性可能發(fā)生變化拾枣,但原實參的值(指向的地址)并沒有發(fā)生變化沃疮。例:
// 設(shè) Student 類對象有 name 和 age 兩個屬性
public static void main(String[] args) {
Student student = new Student("張三",18);
System.out.println("原對象的hashCode值:"+student.hashCode());
referenceTest(student);
System.out.println("測試后的原對象:"+student);
System.out.println("測試后原對象的hashCode值:"+student.hashCode());
}
public static void referenceTest(Student stu){
System.out.println("改變前形參指向?qū)ο蟮膆ashCode值:"+stu.hashCode());
stu.setName("李四");
System.out.println("改變后形參指向的對象:"+stu);
System.out.println("改變后形參指向?qū)ο蟮膆ashCode值:"+stu.hashCode());
}
運行結(jié)果:(這里 hashCode 值可代表地址)
原對象的hashCode值:21685669
改變前形參指向?qū)ο蟮膆ashCode值:21685669
改變后形參指向的對象:Student{name='李四', age=18}
改變后形參指向?qū)ο蟮膆ashCode值:21685669
測試后的原對象:Student{name='李四', age=18}
測試后原對象的hashCode值:21685669
2.如果該形參指向其它對象盒让,或者指向新 new 的對象,那么形參值變?yōu)樾聦ο蟮膬?nèi)存地址司蔬。此后通過該形參操作的會是新的對象邑茄,原實參的值(指向的地址)不會發(fā)生變化。例:
// 設(shè) Student 類對象有 name 和 age 兩個屬性
public static void main(String[] args) {
Student student = new Student("張三",18);
System.out.println("原對象的hashCode值:"+student.hashCode());
referenceTest(student);
System.out.println("測試后的原對象:"+student);
System.out.println("測試后原對象的hashCode值:"+student.hashCode());
}
public static void referenceTest(Student stu){
System.out.println("改變前形參指向?qū)ο蟮膆ashCode值:"+stu.hashCode());
stu = new Student("王五",21);
System.out.println("改變后形參指向的對象:"+stu);
System.out.println("改變后形參指向?qū)ο蟮膆ashCode值:"+stu.hashCode());
}
運行結(jié)果:(這里 hashCode 值可代表地址)
原對象的hashCode值:21685669
改變前形參指向?qū)ο蟮膆ashCode值:21685669
改變后形參指向的對象:Student{name='王五', age=21}
改變后形參指向?qū)ο蟮膆ashCode值:2133927002 // 有變化
測試后的原對象:Student{name='張三', age=18}
測試后原對象的hashCode值:21685669
即俊啼,此后在方法中對形參的操作并不會影響到原來的實參值(原對象的地址)肺缕。
參考博文:
- https://mp.weixin.qq.com/s/F7Niaa7nD1tLApCEGKAj4A
- https://juejin.im/post/5bce68226fb9a05ce46a0476
- https://zhuanlan.zhihu.com/p/29074454
如有問題,歡迎交流~
歡迎您的點贊授帕、收藏和評論同木!