java方法及this關鍵字原理分析詳解

java類的方法,我特別喜歡《java編程思想》裡面的描述,這本書說java類之間的相互通信是通過消息。比如顧客類的對象調用一個eat方法,可以看成是我們向顧客類發送瞭一條消息,命令他去吃飯。

步驟1 、給顧客增加一個吃飯的方法

假設我們有一個顧客類:

class Customer {
	String name;		// 客戶姓名
	String sex;		//性別
	String birthDate;	//生日
	String phoneNumber;	//電話號碼
        int status;	//客戶狀態(1:正常,2:不正常)
        void eat(){
           System.out.println("eating...");
        }
  
}

void代表方法沒有返回值。

步驟 2 、 沒有加static的屬性和方法,一定需要先new對象

這是一句口訣,new的意思是新建一個對象,Customer類隻是一個聲明式的東西,不是真實存在的對象。
隻要你沒有加static,就必須要在運行的時候創建瞭對象,才可以使用!

說得武斷一點,隻要你沒有加static,你要運行eat方法,就必須要new一個對象。

至於為什麼,現在先不管,就這麼強記一下吧。

步驟 3 、 用new出來的對象去執行eat方法

class Demo {
   public static void main(String[] args){
		Customer c = new Customer();
                c.eat();  
   }
}

運行,就可以看到eat方法被調用瞭。

步驟 4 、 怎麼理解c.eat()

c是顧客對象的一個引用,或者叫指針。c就是一個標記,本身不會保存顧客對象的數據,但是我們操作c,就和操作顧客對象是一樣的。

這行代碼代表我們向這個顧客發送瞭一條吃飯消息,顧客很聰明的,一下子就懂瞭,於是他就用一個句點 . 加上eat(方法名),再打一個圓括號的方式,執行瞭這個方法。

這是一次成功的通訊。

步驟 5 、 消息接受器

一個類裡面的方法,我們可以把它稱為​​消息接收器,這樣非常有利於理解。

步驟 6 、 如果有兩個顧客?

問個問題,如果有兩個顧客,一個叫亞瑟,一個叫安琪拉。那麼他們調用的eat方法是相同的還是不同的。

如果這個出面試題,是很有區分度的。

步驟 7 、 答案

答案是相同的方法,沒想到吧。
原因是類加載器把class文件加載到虛擬機,內存的劃分會把類的所有方法都放到一個叫做方法區的地方。不管你new多少個對象,調用的都是同一個方法。因為如果你每new一個對象,都專門開辟一個內存空間來存放方法的話,未免也太浪費瞭。

現在就有一個新的問題,既然方法是同一份,我怎麼知道是誰呢?

步驟 8 、其實有個this

事情的真相是這樣的…
Demo.java被javac命令編譯成Demo.class和Customer.class,然後運行的時候,首先是類加載器去加載Demo.class,發現還用到瞭Customer.class,於是順便把Customer.class也加載瞭。

在這個例子中,JVM裡面有兩個地方需要我們關註,一個是堆,一個是方法區。

什麼是堆?堆是一塊存放對象的區域,你每次new一個對象,這個對象就是放在堆裡面。比如有這樣的代碼:

class Demo {
   public static void main(String[] args){
		Customer c1 = new Customer();
		Customer c2 = new Customer();
		c1.name = "亞瑟";
		c2.name = "安其拉";
   }
}

我們new瞭兩個對象,其中c1和c2是他們的引用,這兩個對象是放在堆裡面的。

在這裡插入圖片描述

而eat方法,是放在方法區,隻有一份。

在這裡插入圖片描述

然後重頭戲來瞭,當你用其中某一個對象去調用eat方法的時候,其實會傳過去一個this。

c1.eat();

this會神不知鬼不覺地作為eat方法的參數,傳入eat方法。
每一個對象都有一個this,this並不是你設置的某個屬性,而是java給每一個類都設置的標記,你察覺不到它的存在,但是它一直都在。

在這裡插入圖片描述

步驟 9 、 在eat方法裡面直接使用this

我們在eat方法裡面直接使用this,就是這樣的:

void eat(){
   System.out.println(this.name + " eating...");
}

效果:

在這裡插入圖片描述

其實,它的底層是這樣的,雖然我們不能這麼寫:

void eat(this){
   System.out.println(this.name + " eating...");
}

我們在方法裡面調用某個變量,這個變量要麼就是參數傳進來的,要麼就是這個類的屬性,這個你同意不?基於這個常理,這個this不是我們定義的屬性,那麼隻能是參數傳進來的。隻不過,這個過程是JVM做的,我們察覺不到。

步驟 10 、 構造方法

每一個類都有構造方法,寫法是這樣的。

public Customer(){
	
}

這是空的構造方法,你不寫也沒事,就算不寫也會有這麼一個方法。構造方法的作用就是在你new對象的時候,直接調用。

記住一個口訣:

1.類名打括號 = 調用構造方法,左邊一定會有個new。
2.方法名打括號 = 調用這個方法,默認傳入一個this(沒有static的情況)

3.方法名打括號 = 調用這個方法,不傳入this(有static的情況)

​現在再回過頭來看new的語句,是不是對上瞭?

Customer c1 = new Customer();

步驟 11 、 總結:this的意義是什麼?

因為方法是獨一份,所有類的對象共享這些方法,JVM為瞭搞清楚到底是誰去調用某個方法,就在每一個類裡面維護一個this。當類被實例化(就是new的時候),這個this就指向這個對象,然後在調用方法的時候,隱式地傳入this,就可以在那個方法中獲取到當前真實對象的數據瞭。

就有點像什麼呢,有點像花名。

什麼叫花名,比如小區的物業會設置一個花名,他們對外會宣稱,管理1-10棟的物業管傢的名字是小芳,這個小芳就是花名。這是一個虛擬的名字,業主也隻知道物業管傢是小芳,但是小芳可能今天是張三,明天是李四。張三李四都離職瞭,王五成為物業管傢,對外的說法也是他叫管傢小芳。

物業設置管傢這個職位,並不知道究竟誰來做,於是先設置好,管傢就叫做小芳。小芳就相當於this,隻有等到張三來任職瞭,這個this就是張三。張三跑路,李四來瞭,這個this就是李四。

步驟 12 、 道理我都懂,那static又是什麼?

我們已經不止一次提到static瞭,static到底是啥啊,別急,下節我們來分享,先把這一節的知識點消化一下吧。

步驟 13 、 本節完整源碼

class Customer {
	String name;		// 客戶姓名
	String sex;		//性別
	String birthDate;	//生日
	String phoneNumber;	//電話號碼
	int status;	        //客戶狀態(1:正常,2:不正常)
	public Customer(){
		
	}
	void eat(){
	   System.out.println(this.name + " eating...");
	}
}
class Vip {
}
class Demo {
   public static void main(String[] args){
		Customer c1 = new Customer();
		Customer c2 = new Customer();
		c1.name = "亞瑟";
		c2.name = "安其拉";
		c1.eat();
   }
}

視頻講解:

Java對this的理解

Java中關於成員方法的講解

以上就是java方法及this關鍵字原理分析詳解的詳細內容,更多關於java方法及this關鍵字的資料請關註WalkonNet其它相關文章!

推薦閱讀: