Java的方法和this關(guān)鍵字如何理解與應(yīng)用
java類(lèi)的方法,我特別喜歡《java編程思想》里面的描述,這本書(shū)說(shuō)java類(lèi)之間的相互通信是通過(guò)消息。比如顧客類(lèi)的對(duì)象調(diào)用一個(gè)eat方法,可以看成是我們向顧客類(lèi)發(fā)送了一條消息,命令他去吃飯。
步驟 1 、給顧客增加一個(gè)吃飯的方法
假設(shè)我們有一個(gè)顧客類(lèi):
class Customer {
String name; // 客戶(hù)姓名
String sex; //性別
String birthDate; //生日
String phoneNumber; //電話(huà)號(hào)碼
int status; //客戶(hù)狀態(tài)(1:正常,2:不正常)
void eat(){
System.out.println("eating...");
}
}
void代表方法沒(méi)有返回值。
步驟 2 、 沒(méi)有加static的屬性和方法,一定需要先new對(duì)象
這是一句口訣,new的意思是新建一個(gè)對(duì)象,Customer類(lèi)只是一個(gè)聲明式的東西,不是真實(shí)存在的對(duì)象。
只要你沒(méi)有加static,就必須要在運(yùn)行的時(shí)候創(chuàng)建了對(duì)象,才可以使用!
說(shuō)得武斷一點(diǎn),只要你沒(méi)有加static,你要運(yùn)行eat方法,就必須要new一個(gè)對(duì)象。
至于為什么,現(xiàn)在先不管,就這么強(qiáng)記一下吧。
步驟 3 、 用new出來(lái)的對(duì)象去執(zhí)行eat方法
class Demo {
public static void main(String[] args){
Customer c = new Customer();
c.eat();
}
}
運(yùn)行,就可以看到eat方法被調(diào)用了。
步驟 4 、 怎么理解c.eat()
c是顧客對(duì)象的一個(gè)引用,或者叫指針。c就是一個(gè)標(biāo)記,本身不會(huì)保存顧客對(duì)象的數(shù)據(jù),但是我們操作c,就和操作顧客對(duì)象是一樣的。
這行代碼代表我們向這個(gè)顧客發(fā)送了一條吃飯消息,顧客很聰明的,一下子就懂了,于是他就用一個(gè)句點(diǎn) . 加上eat(方法名),再打一個(gè)圓括號(hào)的方式,執(zhí)行了這個(gè)方法。
這是一次成功的通訊。
步驟 5 、 消息接受器
一個(gè)類(lèi)里面的方法,我們可以把它稱(chēng)為消息接收器,這樣非常有利于理解。
步驟 6 、 如果有兩個(gè)顧客?
問(wèn)個(gè)問(wèn)題,如果有兩個(gè)顧客,一個(gè)叫亞瑟,一個(gè)叫安琪拉。那么他們調(diào)用的eat方法是相同的還是不同的。
如果這個(gè)出面試題,是很有區(qū)分度的。
步驟 7 、 答案
答案是相同的方法,沒(méi)想到吧。
原因是類(lèi)加載器把class文件加載到虛擬機(jī),內(nèi)存的劃分會(huì)把類(lèi)的所有方法都放到一個(gè)叫做方法區(qū)的地方。不管你new多少個(gè)對(duì)象,調(diào)用的都是同一個(gè)方法。因?yàn)槿绻忝縩ew一個(gè)對(duì)象,都專(zhuān)門(mén)開(kāi)辟一個(gè)內(nèi)存空間來(lái)存放方法的話(huà),未免也太浪費(fèi)了。
現(xiàn)在就有一個(gè)新的問(wèn)題,既然方法是同一份,我怎么知道是誰(shuí)呢?
步驟 8 、其實(shí)有個(gè)this
事情的真相是這樣的…
Demo.java被javac命令編譯成Demo.class和Customer.class,然后運(yùn)行的時(shí)候,首先是類(lèi)加載器去加載Demo.class,發(fā)現(xiàn)還用到了Customer.class,于是順便把Customer.class也加載了。
在這個(gè)例子中,JVM里面有兩個(gè)地方需要我們關(guān)注,一個(gè)是堆,一個(gè)是方法區(qū)。
什么是堆?堆是一塊存放對(duì)象的區(qū)域,你每次new一個(gè)對(duì)象,這個(gè)對(duì)象就是放在堆里面。比如有這樣的代碼:
class Demo {
public static void main(String[] args){
Customer c1 = new Customer();
Customer c2 = new Customer();
c1.name = "亞瑟";
c2.name = "安其拉";
}
}
我們new了兩個(gè)對(duì)象,其中c1和c2是他們的引用,這兩個(gè)對(duì)象是放在堆里面的。

而eat方法,是放在方法區(qū),只有一份。

然后重頭戲來(lái)了,當(dāng)你用其中某一個(gè)對(duì)象去調(diào)用eat方法的時(shí)候,其實(shí)會(huì)傳過(guò)去一個(gè)this。
c1.eat();
this會(huì)神不知鬼不覺(jué)地作為eat方法的參數(shù),傳入eat方法。
每一個(gè)對(duì)象都有一個(gè)this,this并不是你設(shè)置的某個(gè)屬性,而是java給每一個(gè)類(lèi)都設(shè)置的標(biāo)記,你察覺(jué)不到它的存在,但是它一直都在。

步驟 9 、 在eat方法里面直接使用this
我們?cè)趀at方法里面直接使用this,就是這樣的:
void eat(){
System.out.println(this.name + " eating...");
}
效果:

其實(shí),它的底層是這樣的,雖然我們不能這么寫(xiě):
void eat(this){
System.out.println(this.name + " eating...");
}
我們?cè)诜椒ɡ锩嬲{(diào)用某個(gè)變量,這個(gè)變量要么就是參數(shù)傳進(jìn)來(lái)的,要么就是這個(gè)類(lèi)的屬性,這個(gè)你同意不?基于這個(gè)常理,這個(gè)this不是我們定義的屬性,那么只能是參數(shù)傳進(jìn)來(lái)的。只不過(guò),這個(gè)過(guò)程是JVM做的,我們察覺(jué)不到。
步驟 10 、 構(gòu)造方法
每一個(gè)類(lèi)都有構(gòu)造方法,寫(xiě)法是這樣的。
public Customer(){
}
這是空的構(gòu)造方法,你不寫(xiě)也沒(méi)事,就算不寫(xiě)也會(huì)有這么一個(gè)方法。構(gòu)造方法的作用就是在你new對(duì)象的時(shí)候,直接調(diào)用。
記住一個(gè)口訣:
1.類(lèi)名打括號(hào) = 調(diào)用構(gòu)造方法,左邊一定會(huì)有個(gè)new。
2.方法名打括號(hào) = 調(diào)用這個(gè)方法,默認(rèn)傳入一個(gè)this(沒(méi)有static的情況)
3.方法名打括號(hào) = 調(diào)用這個(gè)方法,不傳入this(有static的情況)
現(xiàn)在再回過(guò)頭來(lái)看new的語(yǔ)句,是不是對(duì)上了?
Customer c1 = new Customer();
步驟 11 、 總結(jié):this的意義是什么?
因?yàn)榉椒ㄊ仟?dú)一份,所有類(lèi)的對(duì)象共享這些方法,JVM為了搞清楚到底是誰(shuí)去調(diào)用某個(gè)方法,就在每一個(gè)類(lèi)里面維護(hù)一個(gè)this。當(dāng)類(lèi)被實(shí)例化(就是new的時(shí)候),這個(gè)this就指向這個(gè)對(duì)象,然后在調(diào)用方法的時(shí)候,隱式地傳入this,就可以在那個(gè)方法中獲取到當(dāng)前真實(shí)對(duì)象的數(shù)據(jù)了。
就有點(diǎn)像什么呢,有點(diǎn)像花名。
什么叫花名,比如小區(qū)的物業(yè)會(huì)設(shè)置一個(gè)花名,他們對(duì)外會(huì)宣稱(chēng),管理1-10棟的物業(yè)管家的名字是小芳,這個(gè)小芳就是花名。這是一個(gè)虛擬的名字,業(yè)主也只知道物業(yè)管家是小芳,但是小芳可能今天是張三,明天是李四。張三李四都離職了,王五成為物業(yè)管家,對(duì)外的說(shuō)法也是他叫管家小芳。
物業(yè)設(shè)置管家這個(gè)職位,并不知道究竟誰(shuí)來(lái)做,于是先設(shè)置好,管家就叫做小芳。小芳就相當(dāng)于this,只有等到張三來(lái)任職了,這個(gè)this就是張三。張三跑路,李四來(lái)了,這個(gè)this就是李四。
步驟 12 、 道理我都懂,那static又是什么?
我們已經(jīng)不止一次提到static了,static到底是啥啊,別急,下節(jié)我們來(lái)分享,先把這一節(jié)的知識(shí)點(diǎn)消化一下吧。
步驟 13 、 本節(jié)完整源碼
class Customer {
String name; // 客戶(hù)姓名
String sex; //性別
String birthDate; //生日
String phoneNumber; //電話(huà)號(hào)碼
int status; //客戶(hù)狀態(tài)(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();
}
}
視頻講解:
https://www.bilibili.com/video/BV1df4y1c7Pe/
https://www.bilibili.com/video/BV1N341127CZ/
到此這篇關(guān)于Java的方法和this關(guān)鍵字如何理解與應(yīng)用的文章就介紹到這了,更多相關(guān)Java 方法 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java springboot壓縮文件上傳,解壓,刪除壓縮包方式
這篇文章主要介紹了Java springboot壓縮文件上傳,解壓,刪除壓縮包方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2025-04-04
在springboot中使用注解將值注入?yún)?shù)的操作
這篇文章主要介紹了在springboot中使用注解將值注入?yún)?shù)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-04-04
SpringBoot使用CXF集成WebService的方法
這篇文章主要介紹了SpringBoot使用CXF集成WebService的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
java調(diào)用mysql存儲(chǔ)過(guò)程實(shí)例分析
這篇文章主要介紹了java調(diào)用mysql存儲(chǔ)過(guò)程的方法,以實(shí)例形式較為詳細(xì)的分析了mysql數(shù)據(jù)庫(kù)的建立和存儲(chǔ)過(guò)程的實(shí)現(xiàn)方法,需要的朋友可以參考下2015-06-06
Spring boot2.x中集成H2數(shù)據(jù)庫(kù)代碼實(shí)例
這篇文章主要介紹了Spring boot2.x中集成H2數(shù)據(jù)庫(kù)代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12
java構(gòu)造函數(shù)的三種類(lèi)型總結(jié)
在本篇文章里小編給大家整理了一篇關(guān)于java構(gòu)造函數(shù)的三種類(lèi)型總結(jié)內(nèi)容,有需要的朋友們可以學(xué)習(xí)參考下。2021-01-01
java查詢(xún)近七日數(shù)據(jù)功能的實(shí)現(xiàn)
這篇文章主要介紹了java查詢(xún)近七日數(shù)據(jù)功能的實(shí)現(xiàn),文章內(nèi)容詳細(xì),簡(jiǎn)單易懂,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2023-01-01
springboot整合apache ftpserver詳細(xì)教程(推薦)
這篇文章主要介紹了springboot整合apache ftpserver詳細(xì)教程,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-01-01

