Java關鍵字---this的由來和其三大作用
【聲明】歡迎轉載,但請保留文章原始出處→_→
秦學苦練:http://www.cnblogs.com/Qinstudy/
文章來源:http://www.cnblogs.com/Qinstudy/p/Qinstudy.html
【正文】
秦學苦練博主,你好!我是立誌要掌握Java編程語言的一名小白,最近我在學習Java中的面向對象的三大特性(封裝、繼承、多態)時,常常會碰到this關鍵詞,但書上只是簡單講了this的幾個作用,我聽得是雲裏霧裏,並沒有講解Java設計者為什麽要創建this關鍵字,是出於一個什麽原因創建了this關鍵字呢?還有this關鍵字解決了什麽問題呢?
這些都是我的疑惑,至於我為什麽問到這些問題,是因為我覺得學習概念不僅僅要學習概念的本身,更重要的是了解、理解知識的由來。再次拜謝博主啦----秦學苦練
Java初學者——小張
秦學苦練的回答:
小張,你好!我剛開始學習this時,也只知道this關鍵字在程序中有哪些作用,但不知道this關鍵字怎麽產生的?你講的這句話很好--學習概念不僅僅要學習概念的本身,更重要的是了解、理解知識的由來
你提出了兩個問題,我把問題羅列出來:
1.Java設計者為什麽要創建this關鍵字,是出於一個什麽原因創建了this關鍵字呢? 簡言之,this關鍵字產生的背景
2.還有this關鍵字解決了什麽問題呢? 簡言之,this關鍵字的作用
受“更重要的是了解、理解知識的由來”的鼓舞,我去網上查閱了相關資料,《Java編程思想》(第四版第5章5.4,P84頁)給出了不錯的回答。下面我把書中的知識按照自己的理解講述一下:
this關鍵字可以這麽去理解,在英語中,指示代詞this,用來指代某個東西,例如:Look,this is a table. this指代table。
在Java語言的設計中,this也是具有指向關系的。
1. this關鍵字產生的背景
我先來回答,Java設計人員為什麽創建了this關鍵字?我將用一段代碼來說明原因。
1 class Person{ 2 public void speak(){ 3 System.out.println("某一個對象去調用方法"); 4 } 5 6 public static void main(String[] args){ 7 Person p1=new Person(); 8 Person p2=new Person(); 9 p1.speak(); //p1對象去調用speak()方法 10 p2.speak(); //p2對象去調用speak()方法
11 }
(親測)運行結果:
某一個對象去調用方法
某一個對象去調用方法
但是,當只有一個speak()方法,編譯器如何知道speak()方法是被p1還是被p2所調用的呢?為了能用簡便、面向對象的語法來編寫代碼--即“發送消息給對象”,編譯器做了一些幕後工作。它暗自把”所操作對象的引用”作為第一個參數傳遞給speak()方法。所以上述調用後的方法變成了這樣:
Person speak(p1);
Person speak(p2);
這是內部的表現形式。如果我們這樣寫,編譯器會報錯,但這種寫法會幫助你去理解編譯器所做的事。假設你希望在方法的內部獲得對當前對象的引用,由於這個引用是編譯器“偷偷”傳入的,所以沒有標識符可用。
於是,為了解決‘在方法的內部獲得對當前對象的引用’這一背景問題,Java設計師專門設計了this關鍵字,用來在方法的內部獲得對當前對象的引用。this關鍵字只能在方法內部使用,表示對"調用方法的那個對象“的引用。可以這麽說,this的用法和其他new出來對象的引用並無不同。
但要註意,如果在方法內部調用同一個類的另一個方法時,可以省略this,直接調用即可。因為當前方法中的this引用會自動應用於同一類中的其他方法。所以小張你可以這樣寫代碼,編譯器是不會報錯的:
class Person{
public void speak(){}
public void eat(){
//在方法內部調用同一個類的另一個方法時,可以省略this,直接調用即可。
speak(); //編譯器自動翻譯成this.speak()。
}
public static void main(String[] args){
}
}
這就是this關鍵字為什麽產生了,小張啊,總而言之,this關鍵字產生的背景:為了解決‘在方法的內部獲得對當前對象的引用’這一問題。this關鍵字被設計出來了,用來在方法的內部獲得對當前對象的引用。
2.this關鍵字的作用
三大作用:
1) 區分局部變量和成員變量同名情況,this.成員變量調用本類中的成員變量;
2)通過this.方法()調用本類中的方法;
3)this在構造方法之間的調用。
小張啊,也許你看了這三大作用,可能還是雲裏霧裏。你在信中,說自己在學習Java中的面向對象的三大特性(封裝、繼承、多態),所以我認為你已經懂了private封裝、extends繼承,並且
知道set、get方法,和空參、有參構造方法的相關概念。下面這個程序需要你有上面的這些基礎,如果你暫時沒有這些基礎,不用著急,先去看看這些基礎。看懂這些基礎後,你再回過頭來,看下面的代碼,你會有種豁然開朗的頓悟!
如果你已經有了上面這些基礎,下面我用一個完整的程序將this關鍵字的三大作用講清楚。
package day_12; //Person類 public class Person { private String name; private int age; public Person(String name){//String name中的name是局部變量 // this的作用1:區分局部變量和成員變量同名情況 this.name=name; //this.name中的name是成員變量 } public Person(String name,int age){ //作用3:this關鍵字在構造方法間的調用 this(name); //用this(name)去調用只有一個參數 name的構造方法 this.age=age; } public void setName(String name){ this.name=name; } public String getName(){ return this.name; } public void setAge(){ this.age=age; } public int getAge(){ return this.age; } }
package day_12; //Worker類 public class Worker extends Person { public Worker(String name,int age){ super(name,age); //super(name,age) 調用父類Person的具有兩個參數的構造方法。 } public void work(){ //作用2:this在方法中的使用 System.out.println(this.getName()+"在砍樹!"); } }
使用this.getName() 有封裝的思想在裏面。由於不知道調用getName()方法的對象是誰,我就用模糊的概念,哪個對象去調用work()方法,this就指代哪個對象。
很顯然,Worker類的對象wk去調用work()方法,那this就指向了wk對象。所以用this.getName()來獲得wk對象中的屬性:光頭強。
//測試類:Test
package day_12; public class Test { public static void main(String[] args){ Worker wk=new Worker("光頭強",28); wk.work(); } }
(親測)運行結果:光頭強在砍樹!
3.方法論
小張,很高興,你能提出這樣兩個問題,這兩個問題促使我去思考this關鍵字產生的背景以及this的三大作用。給你講述了this之後,我覺得自己對this的理解又加深了!小張啊,以後在Java學習過程中遇到雲裏霧裏的問題,盡管給我寫信,只要是我能弄明白的,一定平鋪直敘地講清楚。學習一門語言,多思考、常總結是不錯的狀態,與你共勉!
Java關鍵字---this的由來和其三大作用