全面理解Java中的引用傳遞和值傳遞
關(guān)于Java傳參時(shí)是引用傳遞還是值傳遞,是一個(gè)討論比較多的話題,
有說Java中只有值傳遞,也有些地方說引用傳遞和值傳遞都存在,本篇記錄思考過程,不保證正確性,
感興趣的同學(xué)一起討論。
1.基本類型和引用類型在內(nèi)存中的保存
Java中數(shù)據(jù)類型分為兩大類,基本類型和對(duì)象類型。相應(yīng)的,變量也有兩種類型:基本類型和引用類型。
基本類型的變量保存原始值,即它代表的值就是數(shù)值本身;
而引用類型的變量保存引用值,"引用值"指向內(nèi)存空間的地址,代表了某個(gè)對(duì)象的引用,而不是對(duì)象本身,
對(duì)象本身存放在這個(gè)引用值所表示的地址的位置。
基本類型包括:byte,short,int,long,char,float,double,Boolean,returnAddress,
引用類型包括:類類型,接口類型和數(shù)組。
相應(yīng)的,變量也有兩種類型:基本類型和引用類型。
2.變量的基本類型和引用類型的區(qū)別
基本數(shù)據(jù)類型在聲明時(shí)系統(tǒng)就給它分配空間:
int a; a=10;//正確,因?yàn)槁暶鱝時(shí)就分配了空間
引用則不同,它聲明時(shí)只給變量分配了引用空間,而不分配數(shù)據(jù)空間:
Date date; //執(zhí)行實(shí)例化,開辟數(shù)據(jù)空間存放Date對(duì)象,然后把空間的首地址傳給today變量 //date=new Date(); //如果注釋掉上一步操作 //The local variable date may not have been initialized //也就是說對(duì)象的數(shù)據(jù)空間沒有分配 date.getDate();
看一下下面的初始化過程,注意"引用"也是占用空間的,一個(gè)空Object對(duì)象的引用大小大概是4byte:
Date a,b; //在內(nèi)存開辟兩個(gè)引用空間 a = new Date();//開辟存儲(chǔ)Date對(duì)象的數(shù)據(jù)空間,并把該空間的首地址賦給a b = a; //將a存儲(chǔ)空間中的地址寫到b的存儲(chǔ)空間中
3.引用傳遞和值傳遞
這里要用實(shí)際參數(shù)和形式參數(shù)的概念來幫助理解,
值傳遞:
方法調(diào)用時(shí),實(shí)際參數(shù)把它的值傳遞給對(duì)應(yīng)的形式參數(shù),函數(shù)接收的是原始值的一個(gè)copy,此時(shí)內(nèi)存中存在兩個(gè)相等的基本類型,即實(shí)際參數(shù)和形式參數(shù),后面方法中的操作都是對(duì)形參這個(gè)值的修改,不影響實(shí)際參數(shù)的值。
引用傳遞:
也稱為傳地址。方法調(diào)用時(shí),實(shí)際參數(shù)的引用(地址,而不是參數(shù)的值)被傳遞給方法中相對(duì)應(yīng)的形式參數(shù),函數(shù)接收的是原始值的內(nèi)存地址;
在方法執(zhí)行中,形參和實(shí)參內(nèi)容相同,指向同一塊內(nèi)存地址,方法執(zhí)行中對(duì)引用的操作將會(huì)影響到實(shí)際對(duì)象。
看一個(gè)例子:
class MyObj{ public int b=99; }
分別傳參int和對(duì)象類型:
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方法中對(duì)obj類做的修改影響了obj這個(gè)對(duì)象。
這里要特殊考慮String,以及Integer、Double等幾個(gè)基本類型包裝類,它們都是immutable類型,
因?yàn)闆]有提供自身修改的函數(shù),每次操作都是新生成一個(gè)對(duì)象,所以要特殊對(duì)待,可以認(rèn)為是和基本數(shù)據(jù)類型相似,傳值操作。
看下面的例子:
public class ReferencePkValue1 { public static void main(String[] args){ ReferencePkValue1 pk=new ReferencePkValue1(); //String類似基本類型,值傳遞,不會(huì)改變實(shí)際參數(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
對(duì)String和StringBuffer的操作產(chǎn)生了不同的結(jié)果。
4.結(jié)論
結(jié)合上面的分析,關(guān)于值傳遞和引用傳遞可以得出這樣的結(jié)論:
(1)基本數(shù)據(jù)類型傳值,對(duì)形參的修改不會(huì)影響實(shí)參;
(2)引用類型傳引用,形參和實(shí)參指向同一個(gè)內(nèi)存地址(同一個(gè)對(duì)象),所以對(duì)參數(shù)的修改會(huì)影響到實(shí)際的對(duì)象;
(3)String, Integer, Double等immutable的類型特殊處理,可以理解為傳值,最后的操作不會(huì)修改實(shí)參對(duì)象。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java面試常見問題---ConcurrentHashMap
ConcurrentHashMap是由Segment數(shù)組結(jié)構(gòu)和HashEntry數(shù)組結(jié)構(gòu)組成。Segment的結(jié)構(gòu)和HashMap類似,是一種數(shù)組和鏈表結(jié)構(gòu),今天給大家普及java面試常見問題---ConcurrentHashMap知識(shí),一起看看吧2021-06-06Springboot如何使用OSHI獲取和操作系統(tǒng)和硬件信息
這篇文章主要介紹了Springboot如何使用OSHI獲取和操作系統(tǒng)和硬件信息問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-10-10Spring Boot多數(shù)據(jù)源及其事務(wù)管理配置方法
本篇文章主要介紹了Spring Boot多數(shù)據(jù)源及其事務(wù)管理配置方法,具有一定的參考價(jià)值,有興趣的可以了解一下。2017-04-04Java面試突擊為什么要用HTTPS及它的優(yōu)點(diǎn)
這篇文章主要介紹了Java面試突擊為什么要用HTTPS及它的優(yōu)點(diǎn),文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-07-07淺析RxJava處理復(fù)雜表單驗(yàn)證問題的方法
這篇文章主要介紹了RxJava處理復(fù)雜表單驗(yàn)證問題的相關(guān)資料,非常不錯(cuò)具有參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06詳解Spring注入集合(數(shù)組、List、Map、Set)類型屬性
這篇文章主要介紹了詳解Spring注入集合(數(shù)組、List、Map、Set)類型屬性,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01