有時會遇到this作為返回值的情況,那麼此時返回的到底是什麼呢? 返回的是調用this所處方法的那個對象的引用,讀起來有點繞口哈,有沒有想起小學語文分析句子成份的試題,哈哈。 一點點分析的話,主幹是“返回的是引用”; 什麼引用呢?“那個對象的引用”; 哪個對象呢?“調用方法的那個對象”; 調用的哪個 ...
有時會遇到this作為返回值的情況,那麼此時返回的到底是什麼呢?
返回的是調用this所處方法的那個對象的引用,讀起來有點繞口哈,有沒有想起小學語文分析句子成份的試題,哈哈。
一點點分析的話,主幹是“返回的是引用”;
什麼引用呢?“那個對象的引用”;
哪個對象呢?“調用方法的那個對象”;
調用的哪個方法呢?“調用的是this所位於的方法”;這樣就清楚了。
再總結一下就是,this作為返回值時,返回的是調用某方法的對象的引用,這個方法正是包含“return this;”這行命令的那個方法;更簡單點說,就是誰調用返回的就是誰。
為了更清楚、直觀的理解問題,下麵以簡單的例子說明。
包human中定義了Person類,代碼如下:
package human; public class Person { String name; int age; public Person() { } public Person(String n, String g) { name = n; gender = g; } //test:this作返回值 Person reThis1() { Person per = new Person("lu","female"); System.out.println("reThis1 per:" + per.name); return this; } Person reThis2() { Person per = reThis1(); System.out.println("reThis2 per:" + per.name); return this; } Person reThis3() { name = "ma"; return this; } static void equRefer(Person per1, Person per2) { if(per1 == per2) System.out.println("per1指向的對象沒有改變,仍與per2指向同一個對象"); else System.out.println("per1指向的對象已改變,與per2指向不同的對象"); System.out.println("per1:" + per1.name); System.out.println("per2:" + per2.name); } public static void main(String[] args) { Person per1 = new Person("liu","female"); Person per2 = per1; per1 = per1.reThis1(); Person.equRefer(per1, per2); per1 = per1.reThis2(); Person.equRefer(per1, per2); System.out.println("調用reThis3之前,per1.name=" + per1.name); per1 = per1.reThis3(); System.out.println("調用reThis3之後,per1.name=" + per1.name); } }
輸出結果如下:
1 reThis1 per:lu 2 per1指向的對象沒有改變,仍與per2指向同一個對象 3 per1:liu 4 per2:liu 5 reThis1 per:lu 6 reThis2 per:liu 7 per1指向的對象沒有改變,仍與per2指向同一個對象 8 per1:liu 9 per2:liu 10 調用reThis3之前,per1.name=liu 11 調用reThis3之後,per1.name=ma
逐句分析執行過程:
(1).第1句:Person per1 = new Person("liu","female");
創建一個Person對象,將name初始化為“liu”,gender初始化為“female”,並讓per1指向該對象。
(2).第2句:Person per2 = per1;
定義一個Person類的對象引用,並與per1指向同一個對象;具體記憶體分配見圖1:
(3).第3句:per1 = per1.reThis1();
由per1調用reThis1()方法,並將返回值賦給per1;reThis1()方法體內定義了一個局部變數per,並創建一個對象,由per指向它;具體記憶體分配見圖2:
緊接著輸出reThis1 per:lu;最後返回this,並把返回的值賦給per1。
(4).第4句:Person.equRefer(per1, per2);
調用equRefer(per1,per2)來驗證per1的值並未改變;根據下麵的輸出結果也可知per1仍與per2指向原來的對象,也就是說此時this的值與per1的值是一致的;也可以說,誰調用的返回的就是誰。
輸出結果:
per1指向的對象沒有改變,仍與per2指向同一個對象
per1:liu
per2:liu
此時的記憶體圖如圖3:
(5).第5句:per1 = per1.reThis2();
per1調用reThis2(),由(4)可推測,此時per1的值也不會變,是由per1調用的this所處的方法,所以返回的也是per1;具體來分析的話,reThis2()定義了一個局部變數per,並給其賦值為reThis1(),也就是說reThis2()調用了reThis1(),由(3)、(4)可推知,此時的記憶體結構是下麵這樣的:
局部變數per指向的對象與per1是一致的,因為調用reThis1的對象是per1所指的對象,所以返回值也是per1。
此處的輸出結果為:
reThis1 per:lu
reThis2 per:liu
(6).第6句:Person.equRefer(per1, per2);
驗證上面的結論,per1指向沒變,此時的記憶體分配圖如圖4所示:
(7).第7、8、9句:
System.out.println("調用reThis3之前,per1.name=" + per1.name);
per1 = per1.reThis3();
System.out.println("調用reThis3之後,per1.name=" + per1.name);
per1調用reThis3();reThis3()中修改了namer的值,由"liu"改為"ma",然後返回this。
調用前後的記憶體結構分別如圖6、圖7所示:
輸出結果:
調用reThis3之前,per1.name=liu
調用reThis3之後,per1.name=ma
再一次驗證了上述的結論。
關於為什麼使用this,我是這麼理解的:由於定義類的時候尚未創建對象,所以不能確定對象到底叫什麼,就用this來統一表示,等到具體調用時就可以知道對象的名字了,然後就用對象的引用替換this;所以為了便於記憶,可以理解成誰調用返回的就是誰。
至於返回this有什麼意義,我的理解是:記返回this的方法為A,可能A的方法體中對對象的屬性做了某些操作或調用了某些方法完成了某些動作,總之,做完這些操作後就把原對象的引用返回,返回後的狀態是對象最新的狀態,可能與對象調用方法A前不同。