全面理解Java中的引用傳遞和值傳遞
關(guān)于Java傳參時是引用傳遞還是值傳遞,是一個討論比較多的話題,
有說Java中只有值傳遞,也有些地方說引用傳遞和值傳遞都存在,本篇記錄思考過程,不保證正確性,
感興趣的同學(xué)一起討論。
1.基本類型和引用類型在內(nèi)存中的保存
Java中數(shù)據(jù)類型分為兩大類,基本類型和對象類型。相應(yīng)的,變量也有兩種類型:基本類型和引用類型。
基本類型的變量保存原始值,即它代表的值就是數(shù)值本身;
而引用類型的變量保存引用值,"引用值"指向內(nèi)存空間的地址,代表了某個對象的引用,而不是對象本身,
對象本身存放在這個引用值所表示的地址的位置。
基本類型包括:byte,short,int,long,char,float,double,Boolean,returnAddress,
引用類型包括:類類型,接口類型和數(shù)組。
相應(yīng)的,變量也有兩種類型:基本類型和引用類型。
2.變量的基本類型和引用類型的區(qū)別
基本數(shù)據(jù)類型在聲明時系統(tǒng)就給它分配空間:
int a; a=10;//正確,因為聲明a時就分配了空間
引用則不同,它聲明時只給變量分配了引用空間,而不分配數(shù)據(jù)空間:
Date date; //執(zhí)行實例化,開辟數(shù)據(jù)空間存放Date對象,然后把空間的首地址傳給today變量 //date=new Date(); //如果注釋掉上一步操作 //The local variable date may not have been initialized //也就是說對象的數(shù)據(jù)空間沒有分配 date.getDate();
看一下下面的初始化過程,注意"引用"也是占用空間的,一個空Object對象的引用大小大概是4byte:
Date a,b; //在內(nèi)存開辟兩個引用空間 a = new Date();//開辟存儲Date對象的數(shù)據(jù)空間,并把該空間的首地址賦給a b = a; //將a存儲空間中的地址寫到b的存儲空間中
3.引用傳遞和值傳遞
這里要用實際參數(shù)和形式參數(shù)的概念來幫助理解,
值傳遞:
方法調(diào)用時,實際參數(shù)把它的值傳遞給對應(yīng)的形式參數(shù),函數(shù)接收的是原始值的一個copy,此時內(nèi)存中存在兩個相等的基本類型,即實際參數(shù)和形式參數(shù),后面方法中的操作都是對形參這個值的修改,不影響實際參數(shù)的值。
引用傳遞:
也稱為傳地址。方法調(diào)用時,實際參數(shù)的引用(地址,而不是參數(shù)的值)被傳遞給方法中相對應(yīng)的形式參數(shù),函數(shù)接收的是原始值的內(nèi)存地址;
在方法執(zhí)行中,形參和實參內(nèi)容相同,指向同一塊內(nèi)存地址,方法執(zhí)行中對引用的操作將會影響到實際對象。
看一個例子:
class MyObj{
public int b=99;
}
分別傳參int和對象類型:
public class ReferencePkValue2 {
public static void main(String[] args) {
ReferencePkValue2 t = new ReferencePkValue2();
int a=99;
t.test1(a);//這里傳遞的參數(shù)a就是按值傳遞
System.out.println(a);
MyObj obj=new MyObj();
t.test2(obj);//這里傳遞的參數(shù)obj就是引用傳遞
System.out.println(obj.b);
}
public void test1(int a){
a=a++;
System.out.println(a);
}
public void test2(MyObj obj){
obj.b=100;
System.out.println(obj.b);
}
}
輸出是:
99
99
100
100
可以看到,int值沒有發(fā)生變化,但是在test2方法中對obj類做的修改影響了obj這個對象。
這里要特殊考慮String,以及Integer、Double等幾個基本類型包裝類,它們都是immutable類型,
因為沒有提供自身修改的函數(shù),每次操作都是新生成一個對象,所以要特殊對待,可以認(rèn)為是和基本數(shù)據(jù)類型相似,傳值操作。
看下面的例子:
public class ReferencePkValue1 {
public static void main(String[] args){
ReferencePkValue1 pk=new ReferencePkValue1();
//String類似基本類型,值傳遞,不會改變實際參數(shù)的值
String test1="Hello";
pk.change(test1);
System.out.println(test1);
//StringBuffer和StringBuilder等是引用傳遞
StringBuffer test2=new StringBuffer("Hello");
pk.change(test2);
System.out.println(test2.toString());
}
public void change(String str){
str=str+"world";
}
public void change(StringBuffer str){
str.append("world");
}
}
輸出是:
Hello
Helloworld
對String和StringBuffer的操作產(chǎn)生了不同的結(jié)果。
4.結(jié)論
結(jié)合上面的分析,關(guān)于值傳遞和引用傳遞可以得出這樣的結(jié)論:
(1)基本數(shù)據(jù)類型傳值,對形參的修改不會影響實參;
(2)引用類型傳引用,形參和實參指向同一個內(nèi)存地址(同一個對象),所以對參數(shù)的修改會影響到實際的對象;
(3)String, Integer, Double等immutable的類型特殊處理,可以理解為傳值,最后的操作不會修改實參對象。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Spring Boot攔截器Interceptor與過濾器Filter詳細(xì)教程(示例詳解)
本文詳細(xì)介紹了SpringBoot中的攔截器(Interceptor)和過濾器(Filter),包括它們的定義、作用范圍、使用場景、實現(xiàn)步驟、執(zhí)行順序、常見問題及解決方案,感興趣的朋友跟隨小編一起看看吧2025-03-03
java題解Leetcode 8字符串轉(zhuǎn)換整數(shù)
這篇文章主要為大家介紹了java題解Leetcode 8字符串轉(zhuǎn)換整數(shù)實現(xiàn),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
win10下定時運(yùn)行與開機(jī)自啟動jar包的方法記錄
這篇文章主要給大家介紹了關(guān)于win10下定時運(yùn)行與開機(jī)自啟動jar包的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11

