溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Java中值傳遞的深度分析

發(fā)布時(shí)間:2020-09-27 18:20:44 來源:腳本之家 閱讀:99 作者:昆明--菜鳥入門 欄目:編程語言

前言

首先說觀點(diǎn):java只有值傳遞沒有引用傳遞

然后再來看看值傳遞與引用傳遞兩者的定義

值傳遞(pass by value)是指在調(diào)用函數(shù)時(shí)將實(shí)際參數(shù)復(fù)制一份傳遞到函數(shù)中,這樣在函數(shù)中如果對參數(shù)進(jìn)行修改,將不會影響到實(shí)際參數(shù)。

引用傳遞(pass by reference)是指在調(diào)用函數(shù)時(shí)將實(shí)際參數(shù)的地址直接傳遞到函數(shù)中,那么在函數(shù)中對參數(shù)所進(jìn)行的修改,將影響到實(shí)際參數(shù)。

這里牢記值傳遞中將實(shí)際參數(shù)復(fù)制一份

然后就是對于參數(shù)類型:值類型 和 引用類型。

結(jié)合起來理解就是:值類型傳遞,java是將其值內(nèi)容復(fù)制一份給形參;對于引用類型傳遞,java是將其地址復(fù)制一份給形參。

下面結(jié)合實(shí)例深入理解為什么java只有值傳遞

package 字符串;

public class 值傳遞 {
 public static void main(String[] args)
 {
 String str1="abc";
 updateStr1(str1);
 System.out.println("main函數(shù)中"+str1);
 }
 public static void updateStr1(String str1)
 {
 str1="cba"; //<注解>
 System.out.println("調(diào)用函數(shù)中"+str1);
 }
 
}

結(jié)果:

Java中值傳遞的深度分析

在這里我們能夠清晰看到我們傳遞的是String類型的對象即(引用類型),并且在調(diào)用函數(shù)中我們修改了str1為cba,如果是引用傳遞那么我們在主函數(shù)打印則應(yīng)該是cba,

但是很遺憾我們在主函數(shù)中仍然打印出來的是abc。所以我們可以說java是值傳遞類型了嗎,答案是不完全的。

接下來再看這一段代碼:

package 字符串;
  

  public class person {   private int age;   public int getAge() { return age; } public void setAge(int age) { this.age = age; }

}
public class 值傳遞2 {
 public static void main(String[] args)
 {
 person p1=new person();
 p1.setAge(10);
 System.out.println("我在主函數(shù)里對p1的年齡屬性賦值為"+p1.getAge());
 setage(p1);
 System.out.println("我再從主函數(shù)里獲取P1的年齡屬性"+p1.getAge());
 }
 
 public static void setage(person p1)
 {
 p1.setAge(18); //不是我們對它的地址進(jìn)行了操作,而是我們對它地址的內(nèi)容進(jìn)行了操作
 System.out.println("我在調(diào)用函數(shù)里對p1的年齡屬性重新賦值為"+p1.getAge());
 }

}

結(jié)果:

Java中值傳遞的深度分析

咦,怎么回事這次也是傳遞的對象(引用類型),為什么這次我們對年齡這個(gè)字段的修改在主函數(shù)同步了呢?

別急,下面我們先來分析這兩個(gè)例子。

首先第一個(gè)類型的例子中,我們傳遞的是String類型的變量,它是一個(gè)特殊的類型的引用變量。

(不可變字符串:編譯器可讓字符串共享,即將各種字符串存放于公共存儲池中,字符串變量是指向其中相應(yīng)位置    --出自《Java核心技術(shù) 卷1》)

出于這句話的理解就是每個(gè)字符串都對應(yīng)一個(gè)地址:我們例一中是將str1的地址復(fù)制給了我們的形參str1,并且形參中str1的地址進(jìn)行了改變指向了“cba”的地址。所以說在主函數(shù)中的str1的地址仍然指向的是“abc”所對應(yīng)的地址。

所以說對于String類型的變量,我們對于給它重新賦值不是改變了它的內(nèi)容,而是改變了它指向字符串的位置。這也就解釋了為什么java中String類型是不可變類型。

而在我們例二中,我們將p1的地址復(fù)制給了我們形參中的p1,此時(shí)他們都指向的內(nèi)存中一塊相同的地址這里存放著相同內(nèi)容,所以我們在調(diào)用函數(shù)對這個(gè)地址中的內(nèi)容進(jìn)行修改時(shí)就會同步到我們主函數(shù)中的p1。所以這個(gè)并不意味著這個(gè)是引用傳遞。

好吧,那怎么才能解釋好Java確實(shí)是值傳遞呢(上面String類型例子是特殊的引用類型不方便解釋)

下面我們通過這個(gè)例子說明:

package 字符串;
public class person {
 private int age;
 public int getAge() {
 return age;
 }
 public void setAge(int age) {
 this.age = age;
 }

}
public class 值傳遞3 {
 public static void main(String[] args) {
 person p1=new person();
 person p2=new person();
 p1.setAge(10);
 p2.setAge(18);
 System.out.println("我在主函數(shù)里對p1的年齡屬性賦值為"+p1.getAge());
 System.out.println("我在主函數(shù)里對p2的年齡屬性賦值為"+p2.getAge());
 swap(p1,p2); 
 System.out.println("************我是主函數(shù)里的分割線***************");
 //我再在主函數(shù)里分別對p1,p2獲取他們的年齡,若為引用傳遞則p1的年齡應(yīng)該為18,p2為10.
 System.out.println("我在主函數(shù)里獲取p1的年齡"+p1.getAge());
 System.out.println("我在主函數(shù)里獲取p1的年齡"+p2.getAge());
 }
 public static void swap(person p1,person p2)
 {
 System.out.println("************我是調(diào)用函數(shù)里的分割線***************");
 person temp=new person();
 temp=p1;
 p1=p2;
 p2=temp;
 System.out.println("我在調(diào)用函數(shù)里交換了p1和p2指向的地址");
 System.out.println("我在調(diào)用函數(shù)里對p1的年齡屬性賦值為"+p1.getAge());
 System.out.println("我在調(diào)用函數(shù)里對p2的年齡屬性賦值為"+p2.getAge());
 
 }

}

結(jié)果:

Java中值傳遞的深度分析

看到?jīng)],這就是充分說明Java是值傳遞的例子。在這個(gè)例子中我們依然傳遞的是person類的對象p1,p2(引用類型),他們將各自的地址復(fù)制一份到了形參p1、p2。

然后我們在調(diào)用函數(shù)中交換了他們的地址,確實(shí)在調(diào)用函數(shù)中他們的age屬性發(fā)生交換。但是再當(dāng)我們在主函數(shù)獲取他們的age時(shí),如果是引用傳遞則應(yīng)該p1的age為18,p2的age為10,

和我們在調(diào)用函數(shù)中打印結(jié)果一致。但是,很遺憾在主函數(shù)中他們的值仍然是p1(10),p2(18)。所以這也充分印證了java是值傳遞。

那么什么是引用傳遞呢?我們把代碼放入C#看看。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace 值傳遞or引用傳遞
{
 public class person
 {
  private int age;
  public int getAge()
  {
   return age;
  }
  public void setAge(int age)
  {
   this.age = age;
  }

 }
 class Program
 {
  static void Main(string[] args)
  {
   person p1 = new person();
   person p2 = new person();
   person p3 = new person();
   p1.setAge(10);
   p2.setAge(18);
   p3.setAge(15);
   Console.WriteLine("我在主函數(shù)里對p1的年齡屬性賦值為" + p1.getAge());
   Console.WriteLine("我在主函數(shù)里對p2的年齡屬性賦值為" + p2.getAge());
   Console.WriteLine("我在主函數(shù)里對p3的年齡屬性賦值為" + p3.getAge());
   swap(ref p1,ref p2,p3);
   Console.WriteLine("************我是主函數(shù)里的分割線***************");
   //我再在主函數(shù)里分別對p1,p2獲取他們的年齡,若為引用傳遞則p1的年齡應(yīng)該為18,p2為10.
   Console.WriteLine("我在主函數(shù)里獲取p1的年齡" + p1.getAge());
   Console.WriteLine("我在主函數(shù)里獲取p2的年齡" + p2.getAge());
   Console.WriteLine("我在主函數(shù)里獲取p3的年齡" + p3.getAge());
  }
  public static void swap(ref person p1,ref person p2, person p3)
  {
   Console.WriteLine("************我是調(diào)用函數(shù)里的分割線***************");
   person temp = new person();
   temp = p1;
   p1 = p2;
   p2 = temp;
   p3.setAge(20);
   Console.WriteLine("我在調(diào)用函數(shù)里交換了p1和p2指向的地址");
   Console.WriteLine("我在調(diào)用函數(shù)里對p1交換地址后年齡為" + p1.getAge());
   Console.WriteLine("我在調(diào)用函數(shù)里對p2交換地址后年齡為" + p2.getAge());
   Console.WriteLine("我在調(diào)用函數(shù)里修改p3年齡為" + p3.getAge());

  }
 }
}

結(jié)果:

Java中值傳遞的深度分析

請注意在C#中如果我們要實(shí)現(xiàn)引用傳遞,請加上關(guān)鍵字ref,否則,它執(zhí)行的原理仍然與我們java中執(zhí)行的機(jī)制一樣,即拷貝一份地址給形參。

如果你還有點(diǎn)暈,不妨我們來看看下面兩張圖。

Java中值傳遞的深度分析

Java中值傳遞的深度分析

 為了方便大家理解把圖畫成這樣,然后關(guān)于java的值傳遞深度分析就到這里。歡迎大家一起討論。(可以打臉/哈哈)

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對億速云的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI