了解java中對(duì)象基礎(chǔ)Object類
一、Object簡(jiǎn)述
源碼注釋:Object類是所有類層級(jí)關(guān)系的Root節(jié)點(diǎn),作為所有類的超類,包括數(shù)組也實(shí)現(xiàn)了該類的方法,注意這里說的很明確,指類層面。
所以在Java中有一句常說的話,一切皆對(duì)象,這話并不離譜。
1、顯式擴(kuò)展
結(jié)論驗(yàn)證
既然Object作為所有類的父級(jí)別的類,則不需要在顯式的添加繼承關(guān)系,Each01編譯期就會(huì)提示移除冗余。
public class Each01 extends Object { public static void main(String[] args) { System.out.println(new Each01().hashCode()+";"+new ObjEa02().hashCode()); } } class ObjEa02 {} class ObjEa03 extends ObjEa02{}
這里Each01與ObjEa02對(duì)象實(shí)例都有Object類中的hashCode方法,這里對(duì)既有結(jié)論的驗(yàn)證。
編譯文件
再?gòu)腏VM編譯層面看下字節(jié)碼文件,是如何加載,使用javap -c命令查看編譯后的文件,注意Jdk版本1.8;
javap -c Each01.class Compiled from "Each01.java" public class com.base.object.each.Each01 { public com.base.object.each.Each01(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object."<init>":()V 4: return } javap -c ObjEa02.class Compiled from "Each01.java" class com.base.object.each.ObjEa02 { com.base.object.each.ObjEa02(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object."<init>":()V 4: return } javap -c ObjEa03.class Compiled from "Each01.java" class com.base.object.each.ObjEa03 extends com.base.object.each.ObjEa02 { com.base.object.each.ObjEa03(); Code: 0: aload_0 1: invokespecial #1 // Method com/base/object/each/ObjEa02."<init>":()V 4: return }
invokespecial命令:可以查看Jvm的官方文檔中的指令說明,調(diào)用實(shí)例化方法,和父類的初始化方法調(diào)用等,這里通過三個(gè)類的層級(jí)關(guān)系,再次說明Object超類不需要顯式繼承,即使顯式聲明但編譯后源碼依舊會(huì)清除冗余。
2、引用與對(duì)象
通常把下面過程稱為:創(chuàng)建一個(gè)object對(duì)象;
Object object = new Object() ;
細(xì)節(jié)描述:聲明對(duì)象引用object;通過new關(guān)鍵字創(chuàng)建對(duì)象并基于默認(rèn)構(gòu)造方法初始化;將對(duì)象引用object指向創(chuàng)建的對(duì)象。
這一點(diǎn)可以基于Jvm運(yùn)行流程去理解,所以當(dāng)對(duì)象一旦失去全部引用時(shí),會(huì)被標(biāo)記為垃圾對(duì)象,在垃圾收集器運(yùn)行時(shí)清理。
接受任意數(shù)據(jù)類型對(duì)象的引用
既然Object作為Java中所有對(duì)象的超類,則根據(jù)繼承關(guān)系的特點(diǎn),以及向上轉(zhuǎn)型機(jī)制,Object可以接受任意數(shù)據(jù)類型對(duì)象的引用,例如在集合容器或者傳參過程,不確定對(duì)象類型時(shí)可以使用Object:
public class Each02 { public static void main(String[] args) { // 向上轉(zhuǎn)型 Object obj01 = new Each02Obj01("java") ; System.out.println(obj01); // 向下轉(zhuǎn)型 Each02Obj01 each02Obj01 = (Each02Obj01)obj01; System.out.println("name="+each02Obj01.getName()); } } class Each02Obj01 { private String name ; public Each02Obj01(String name) { this.name = name; } @Override public String toString() { return "Each02Obj01{" +"name='" + name +'}'; } public String getName() { return name; } }
這里要強(qiáng)調(diào)一下這個(gè)向上轉(zhuǎn)型的過程:
Object obj01 = new Each02Obj01("java") ;
通過上面流程分析,這里創(chuàng)建一個(gè)父類引用obj01,并指向子類Each02Obj01對(duì)象,所以在輸出的時(shí)候,調(diào)用的是子類的toString方法。
二、基礎(chǔ)方法
1、getClass
在程序運(yùn)行時(shí)獲取對(duì)象的實(shí)例類,進(jìn)而可以獲取詳細(xì)的結(jié)構(gòu)信息并進(jìn)行操作:
public final native Class<?> getClass();
該方法在泛型,反射,動(dòng)態(tài)代理等機(jī)制中有很多場(chǎng)景應(yīng)用。
2、toString
返回對(duì)象的字符串描述形式,Object提供的是類名與無符號(hào)十六進(jìn)制的哈希值組合表示,為了能返回一個(gè)信息明確的字符串,子類通常會(huì)覆蓋該方法:
public String toString() { return getClass().getName()+"@"+Integer.toHexString(hashCode()); }
在Java中,打印對(duì)象的時(shí)候,會(huì)執(zhí)行String.valueOf轉(zhuǎn)換為字符串,該方法的底層依舊是對(duì)象的toString方法:
public void println(Object x) { String s = String.valueOf(x); } public static String valueOf(Object obj) { return (obj == null) ? "null" : obj.toString(); }
3、equals與hashCode
- equals:判斷兩個(gè)對(duì)象是否相等;
- hashCode:返回對(duì)象的哈希碼值;
public native int hashCode(); public boolean equals(Object obj) { return (this == obj); }
equals判斷方法需要考量實(shí)際的場(chǎng)景與策略,例如常見的公民注冊(cè)后分配的身份ID是不能修改的,但是名字可以修改,那么就可能存在這樣的場(chǎng)景:
EachUser eachUser01 = new EachUser(1,"A") ; EachUser eachUser02 = new EachUser(1,"B") ; class EachUser { private Integer cardId ; private String name ; }
從程序本身看,這確實(shí)是創(chuàng)建兩個(gè)對(duì)象,但是放在場(chǎng)景下,這的確是描述同一個(gè)人,所以這時(shí)候可以在equals方法中定義比較規(guī)則,如果ID相同則視為同一個(gè)對(duì)象:
@Override public boolean equals(Object obj) { if (obj != null){ EachUser compareObj = (EachUser)obj ; return this.cardId.intValue()==compareObj.cardId ; } return Boolean.FALSE ; }
這里還要注意值類型和引用類型的區(qū)別,如果出現(xiàn)null比較情況,要返回false。
通常在子類中會(huì)同時(shí)覆蓋這兩個(gè)方法,這樣做法在集合容器的設(shè)計(jì)上已經(jīng)體現(xiàn)的淋漓盡致。
4、thread相關(guān)
- wait:線程進(jìn)入waiting等待狀態(tài),不會(huì)爭(zhēng)搶鎖對(duì)象
- notify:隨機(jī)通知一個(gè)在該對(duì)象上等待的線程;
- notifyAll:?jiǎn)拘言谠搶?duì)象上所有等待的線程;
public final native void wait(long timeout) throws InterruptedException; public final native void notify(); public final native void notifyAll();
注意這里:native關(guān)鍵字修飾的方法,即調(diào)用的是原生函數(shù),也就是常說的基于C/C++實(shí)現(xiàn)的本地方法,以此提高和系統(tǒng)層面的交互效率降低交互復(fù)雜程度。
5、clone
返回當(dāng)前對(duì)象的拷貝:
protected native Object clone() throws CloneNotSupportedException;
關(guān)于該方法的細(xì)節(jié)規(guī)則極度復(fù)雜,要注意下面幾個(gè)核心點(diǎn):
對(duì)象必須實(shí)現(xiàn)Cloneable接口才可以被克??;
數(shù)據(jù)類型:值類型,String類型,引用類型;
深淺拷貝的區(qū)別和與之對(duì)應(yīng)的實(shí)現(xiàn)流程;
在復(fù)雜的包裝類型中,組合的不同變量類型;
6、finalize
當(dāng)垃圾收集器確認(rèn)該對(duì)象上沒有引用時(shí),會(huì)調(diào)用finalize方法,即清理內(nèi)存釋放資源:
protected void finalize() throws Throwable { }
通常子類不會(huì)覆蓋該方法,除非在子類中有一些其他必要的資源清理動(dòng)作。
三、生命周期
1、作用域
在下面main方法執(zhí)行結(jié)束之后,無法再訪問Each05Obj01的實(shí)例對(duì)象,因?yàn)閷?duì)象的引用each05丟失:
public class Each05 { public static void main(String[] args) { Each05Obj01 each05 = new Each05Obj01 (99) ; System.out.println(each05); } }
這里就會(huì)存在一個(gè)問題,引用丟失導(dǎo)致對(duì)象無法訪問,但是對(duì)象在此時(shí)可能還是存在的,并沒有釋放內(nèi)存的占用。
2、垃圾回收機(jī)制
Java通過new創(chuàng)建的對(duì)象會(huì)在堆中開辟內(nèi)存空間存儲(chǔ),當(dāng)對(duì)象失去所有引用時(shí)會(huì)被標(biāo)記為垃圾對(duì)象,進(jìn)而被回收;
這里涉及下面幾個(gè)關(guān)鍵點(diǎn):
- Jvm中垃圾收集器會(huì)監(jiān)控創(chuàng)建的對(duì)象 ;
- 當(dāng)判斷對(duì)象不存在引用時(shí),會(huì)執(zhí)行清理動(dòng)作;
- 完成對(duì)象清理后會(huì)重新整理內(nèi)存空間;
這里存在一個(gè)很難理解的概念,即對(duì)象不存在引用的判斷,也就是常說的可達(dá)性分析算法:基于對(duì)象到根對(duì)象的引用鏈?zhǔn)欠窨蛇_(dá)來判斷對(duì)象是否可以被回收;GC-Roots根引用集合,也可以變相理解為存活對(duì)象的集合。(詳見JVM系列)
通過Object對(duì)象的分析,結(jié)合Java方方面面的機(jī)制和設(shè)計(jì),可以去意會(huì)一些所謂的編程思想。
四、源代碼地址
GitEE·地址
https://gitee.com/cicadasmile/java-base-parent
Wiki·地址
https://gitee.com/cicadasmile/butte-java-note/wikis
到此這篇關(guān)于了解java中對(duì)象基礎(chǔ)Object類的文章就介紹到這了,更多相關(guān)java object類內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring Security CsrfFilter過濾器用法實(shí)例
這篇文章主要介紹了Spring Security CsrfFilter過濾器用法實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11Java設(shè)計(jì)模式之代理模式與裝飾模式實(shí)例詳解
這篇文章主要介紹了Java設(shè)計(jì)模式之代理模式與裝飾模式,結(jié)合具體實(shí)例形式分析了代理模式與裝飾模式的概念、功能、java實(shí)現(xiàn)方法與相關(guān)使用注意事項(xiàng),需要的朋友可以參考下2017-09-09jfinal中stateless模式嵌入shiro驗(yàn)證的實(shí)現(xiàn)方式
這篇文章主要介紹了jfinal中stateless模式嵌入shiro驗(yàn)證,今天,我們就來嘗試一種通過攔截器來實(shí)現(xiàn)的Stateless Jfinal嵌入方式,需要的朋友可以參考下2022-06-06應(yīng)用啟動(dòng)數(shù)據(jù)初始化接口CommandLineRunner和Application詳解
這篇文章主要介紹了應(yīng)用啟動(dòng)數(shù)據(jù)初始化接口CommandLineRunner和Application詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12MyBatis-Plus updateById不更新null值的方法解決
用Mybatis-Plus的updateById()來更新數(shù)據(jù)時(shí),無法將字段設(shè)置為null值,更新后數(shù)據(jù)還是原來的值,本文就來詳細(xì)的介紹一下解決方法,具有一定的參考價(jià)值,感興趣的可以了解一下2023-08-08MyBatis Plus實(shí)現(xiàn)一對(duì)多的查詢場(chǎng)景的三種方法
MyBatis Plus提供了多種簡(jiǎn)便的方式來進(jìn)行一對(duì)多子查詢,本文主要介紹了MyBatis Plus實(shí)現(xiàn)一對(duì)多的查詢場(chǎng)景的三種方法,具有一定的參考價(jià)值,感興趣的可以了解一下2024-07-07