程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> Java關鍵字this(動力節點Java學院整理)

Java關鍵字this(動力節點Java學院整理)

編輯:關於JAVA

Java關鍵字this(動力節點Java學院整理)。本站提示廣大學習愛好者:(Java關鍵字this(動力節點Java學院整理))文章只能為提供參考,不一定能成為您想要的結果。以下是Java關鍵字this(動力節點Java學院整理)正文


我們通常在用Java中的this關鍵字的時候,都知道this是代表正在調用這個類的方法的當前實例。通常情況下理解this關鍵字還是很容易的,但是在我初學的時候,有一個疑問卻一直不能很清晰的理解,現在慢慢的理解了,就想把它記下來,也許有人和我有相同的疑問,說不定可以幫助到別人。我們還是先簡單的看看通常情況下this的作用吧。比如下面的代碼:

public class Leaf {
 private int i = 0;
 Leaf increment() {
  i++;
  return this;
 }
 void print() {
  System.out.println("i = " + i);
 }
 public static void main(String[] args) {
  Leaf x = new Leaf();
  x.increment().increment().print();
 }
}

在Leaf類的main方法中,我們new了一個Leaf實例 x,然後x實例調用increment()方法。如果increment()是普通的方法或者void方法,這個地方就沒有什麼值得我們研究的了。特殊的是,在increment()方法中,我們return的是一個this,這個this代表的就是我們剛剛創建的x。因為x正在調用increment()方法,所以,increment()方法this就很明顯代表的是Leaf的x實例了。

     這看起來沒有什麼可討論的,this就是代表的調用該方法的實例x。可是,假如我們把main()函數修改成下面的樣子

public static void main(String[] args) {
 Leaf x = new Leaf();
 x.increment().increment().print();
  
 Leaf y = new Leaf();
 y.increment().increment().print();
}

以上修改的代碼中,我們增加創建了一個Leaf實例y,然後y也連續調用調用了兩次increment()。現在問題來了,假如 x,y同時調用的increment()方法,那麼this到底能代表誰呢?你可能會覺得這有什麼問題,x調用increment()方法,this就代表x, y調用increment()方法,this就代表y。可問題是,當我們講調用方法的時候,在jvm層面上是找到Leaf類中increment()方法所在的內存地址,然後在java虛擬機棧中創建棧幀.

然後在棧幀中執行方法裡面的代碼。現在看到了吧,也就是說,在jvm執行方法層面,沒有所謂的x調用,y調用了,那麼,方法中的this到底是怎麼確定指向哪個實例的呢?

我們還是來看看Leaf類字節碼中是怎麼展示的,是不是我們漏了什麼,如果我們沒有把x實例或者y實例傳遞到方法裡面去,那麼,在jvm執行方法的時候,是不可能知道this具體指向哪個實例的。

到這裡,我們看到在increment()方法中,編碼中沒有參數,但是在字節碼裡面卻顯示參數個數為1,仔細想想,結果已經很明顯了:jvm在執行編譯的時候,在實例方法中,會默認隱藏的傳遞一個參數,這個參數就是當前調用的實例本身。比如x調用,隱藏就把x傳過去,y調用,就把y傳過去。所以,我們的this才能在jvm執行方法層面確定到底指向的是誰。

上面的結論是我們自己推斷出來的,有沒有那本書對這個有詳細的描述呢?《java編程思想》裡面,對這塊是這樣描述的:

假定我們在一個方法的內部,並希望獲得當前對象的句柄。由於那個句柄是由編譯器“秘密”傳遞的,所以沒有標識符可用。然而,針對這一目的有個專用的關鍵字:this。

在裡面講的這個編譯器秘密傳遞的句柄,就是我們這裡的這個隱藏參數。

    到此為止,關於this的描述想必已經很清楚了,我們在jvm層面對它進行了理解。那麼,各位有沒有興趣在看下下面的這個例子,想想這個基類B中的this代表了什麼呢?

public class B {
 public B() {
  System.out.println(this.getClass().getSimpleName()); 
  System.out.println(((A) this).a); 
 }
}
public class A extends B {
 public int a = 100;  
 public A() {
  a = 200;
 } 
 public static void main(String[] args) {
  new A();
 }
}

這個例子原本是為了了解java具有繼承結構的時候類是怎麼完成初始化的,可是這裡面的B類中的構造函數比較特殊:B類中的構造函數中的this輸出的SimpleName是A。通常我們遇到的情況,B類中的this輸出的SimpleName應該是B,可是這裡卻是A?為什麼?

在上面我們講this的過程中,其實已經涉及到這塊了,在調用java方法創建棧幀的時候,jvm會秘密的傳遞一個當前實例。所以,當我們在執行A的構造函數的時候,默認會調用父類B的構造函數,在調用父類B構造函數的時候,秘密的傳進去的當前實例是 A的實例----因為是在A的構造函數中調用的B,所以,這個地方的this反而代表了A。

以上所述是小編給大家介紹的Java關鍵字this(動力節點Java學院整理),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對網站的支持!

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved