Java關鍵字this(動力節(jié)點Java學院整理)
我們通常在用Java中的this關鍵字的時候,都知道this是代表正在調(diào)用這個類的方法的當前實例。通常情況下理解this關鍵字還是很容易的,但是在我初學的時候,有一個疑問卻一直不能很清晰的理解,現(xiàn)在慢慢的理解了,就想把它記下來,也許有人和我有相同的疑問,說不定可以幫助到別人。我們還是先簡單的看看通常情況下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實例調(diào)用increment()
方法。如果increment()是普通的方法或者void方法,這個地方就沒有什么值得我們研究的了。特殊的是,在increment()
方法中,我們return的是一個this,這個this代表的就是我們剛剛創(chuàng)建的x。因為x正在調(diào)用increment()方法,所以,increment()
方法this就很明顯代表的是Leaf的x實例了。
這看起來沒有什么可討論的,this就是代表的調(diào)用該方法的實例x??墒牵偃缥覀儼裮ain()函數(shù)修改成下面的樣子
public static void main(String[] args) { Leaf x = new Leaf(); x.increment().increment().print(); Leaf y = new Leaf(); y.increment().increment().print(); }
以上修改的代碼中,我們增加創(chuàng)建了一個Leaf實例y,然后y也連續(xù)調(diào)用調(diào)用了兩次increment()?,F(xiàn)在問題來了,假如 x,y同時調(diào)用的increment()方法,那么this到底能代表誰呢?你可能會覺得這有什么問題,x調(diào)用increment()方法,this就代表x, y調(diào)用increment()方法,this就代表y??蓡栴}是,當我們講調(diào)用方法的時候,在jvm層面上是找到Leaf類中increment()
方法所在的內(nèi)存地址,然后在java虛擬機棧中創(chuàng)建棧幀.
然后在棧幀中執(zhí)行方法里面的代碼?,F(xiàn)在看到了吧,也就是說,在jvm執(zhí)行方法層面,沒有所謂的x調(diào)用,y調(diào)用了,那么,方法中的this到底是怎么確定指向哪個實例的呢?
我們還是來看看Leaf類字節(jié)碼中是怎么展示的,是不是我們漏了什么,如果我們沒有把x實例或者y實例傳遞到方法里面去,那么,在jvm執(zhí)行方法的時候,是不可能知道this具體指向哪個實例的。
到這里,我們看到在increment()
方法中,編碼中沒有參數(shù),但是在字節(jié)碼里面卻顯示參數(shù)個數(shù)為1,仔細想想,結(jié)果已經(jīng)很明顯了:jvm在執(zhí)行編譯的時候,在實例方法中,會默認隱藏的傳遞一個參數(shù),這個參數(shù)就是當前調(diào)用的實例本身。比如x調(diào)用,隱藏就把x傳過去,y調(diào)用,就把y傳過去。所以,我們的this才能在jvm執(zhí)行方法層面確定到底指向的是誰。
上面的結(jié)論是我們自己推斷出來的,有沒有那本書對這個有詳細的描述呢?《java編程思想》里面,對這塊是這樣描述的:
假定我們在一個方法的內(nèi)部,并希望獲得當前對象的句柄。由于那個句柄是由編譯器“秘密”傳遞的,所以沒有標識符可用。然而,針對這一目的有個專用的關鍵字:this。
在里面講的這個編譯器秘密傳遞的句柄,就是我們這里的這個隱藏參數(shù)。
到此為止,關于this的描述想必已經(jīng)很清楚了,我們在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具有繼承結(jié)構(gòu)的時候類是怎么完成初始化的,可是這里面的B類中的構(gòu)造函數(shù)比較特殊:B類中的構(gòu)造函數(shù)中的this輸出的SimpleName是A。通常我們遇到的情況,B類中的this輸出的SimpleName應該是B,可是這里卻是A?為什么?
在上面我們講this的過程中,其實已經(jīng)涉及到這塊了,在調(diào)用java方法創(chuàng)建棧幀的時候,jvm會秘密的傳遞一個當前實例。所以,當我們在執(zhí)行A的構(gòu)造函數(shù)的時候,默認會調(diào)用父類B的構(gòu)造函數(shù),在調(diào)用父類B構(gòu)造函數(shù)的時候,秘密的傳進去的當前實例是 A的實例----因為是在A的構(gòu)造函數(shù)中調(diào)用的B,所以,這個地方的this反而代表了A。
以上所述是小編給大家介紹的Java關鍵字this(動力節(jié)點Java學院整理),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關文章
BUUCTF-easy java WEB-INF/web.xml泄露漏洞及其利用方式
這篇文章主要介紹了BUUCTF-easy java WEB-INF/web.xml泄露漏洞及其利用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-07-07SpringBoot+logback默認日志的配置和使用方式
這篇文章主要介紹了SpringBoot+logback默認日志的配置和使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-05-05java實現(xiàn)圖片角度旋轉(zhuǎn)并獲得圖片信息
這篇文章主要為大家詳細介紹了java實現(xiàn)圖片角度旋轉(zhuǎn)并獲得圖片信息,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-02-02使用Spring Boot實現(xiàn)操作數(shù)據(jù)庫的接口的過程
本文給大家分享使用Spring Boot實現(xiàn)操作數(shù)據(jù)庫的接口的過程,包括springboot原理解析及實例代碼詳解,感興趣的朋友跟隨小編一起看看吧2021-07-07