java反射:關於.class和getClass()對象的理解

最明顯的區別就是.getClass()是一個對象實例的方法,只有對象實例纔有這個方法,具體的類是沒有的。類的Class類實例是通過.class獲得的,顯然,類沒有.getClass()方法。不過要談到兩者的區別,還要從Java反射說起。

Java反射學習

所謂反射,可以理解爲在運行時期獲取對象類型信息的操作。傳統的編程方法要求程序員在編譯階段決定使用的類型,但是在反射的幫助下,編程人員可以動態獲取這些信息,從而編寫更加具有可移植性的代碼。嚴格地說,反射並非編程語言的特性,因爲在任何一種語言都可以實現反射機制,但是如果編程語言本身支持反射,那麼反射的實現就會方便很多。

1,獲得類型類

我們知道在Java中一切都是對象,我們一般所使用的對象都直接或間接繼承自Object類。Object類中包含一個方法名叫getClass,利用這個方法就可以獲得一個實例的類型類。類型類指的是代表一個類型的類,因爲一切皆是對象,類型也不例外,在Java使用類型類來表示一個類型。所有的類型類都是Class類的實例。例如,有如下一段代碼:

 


 
  1. A a = new A();

  2. if(a.getClass()==A.class) {

  3. System.out.println("equal");

  4. } else {

  5. System.out.println("unequal");

  6. }

  7. 輸出equal;

可以看到,對象a是A的一個實例,A是某一個類,在if語句中使用a.getClass()返回的結果正是A的類型類,在Java中表示一個特定類型的類型類可以用“類型.class”的方式獲得,因爲a.getClass()獲得是A的類型類,也就是A.class,因此上面的代碼執行的結果就是打印出“equal”。特別注意的是,類型類是一一對應的,父類的類型類和子類的類型類是不同的,因此,假設A是B的子類,那麼如下的代碼將得到“unequal”的輸出:


 
  1. A a = new A();

  2. if(a.getClass()==B.class) {

  3. System.out.println("equal");

  4. } else {

  5. System.out.println("unequal");

  6. }

  7. 輸出unequal;

因此,如果你知道一個實例,那麼你可以通過實例的“getClass()”方法獲得該對象的類型類,如果你知道一個類型,那麼你可以使用“.class”的方法獲得該類型的類型類。

2,獲得類型的信息

在獲得類型類之後,你就可以調用其中的一些方法獲得類型的信息了,主要的方法有:

getName():String:獲得該類型的全稱名稱。

getSuperClass():Class:獲得該類型的直接父類,如果該類型沒有直接父類,那麼返回null。

getInterfaces():Class[]:獲得該類型實現的所有接口。

isArray():boolean:判斷該類型是否是數組。

isEnum():boolean:判斷該類型是否是枚舉類型。

isInterface():boolean:判斷該類型是否是接口。

isPrimitive():boolean:判斷該類型是否是基本類型,即是否是int,boolean,double等等。

isAssignableFrom(Classcls):boolean:判斷這個類型是否是類型cls的父(祖先)類或父(祖先)接口。

getComponentType():Class:如果該類型是一個數組,那麼返回該數組的組件類型。

此外還可以進行類型轉換這類的操作,主要方法有:

asSubclass(Class clazz):Class:將這個類型


 

還有從其他資料上找到的一下內容一併分享:

1、出現的時期不同:Class.forName(),getClass()在運行時加載;Class.class是在編譯器加載,即.class是靜態加載,.getClass()是動態加載。這裏有些個疑問?Class.forName("XXX")這方法是動態加載class,先把類文件加載進來,再使用.newInstance()時創建了一個對象。

new ClassName(),就是所謂的靜態加載,

Class.forName("ClassName"),就是所謂的動態加載。

區別在於“靜態加載”的類在編譯的時候就要提供,而動態加載的類在源程序編譯時可以缺席。

Class.forName(xxx.xx.xx) 返回的是一個類, .newInstance() 後才創建一個對象 Class.forName(xxx.xx.xx);的作用是要求JVM查找並加載指定的類,也就是說JVM會執行該類的靜態代碼段。

2、舉個例子,Iterator it = s.iterator();得到的it的真正類型是KeyIterator,是Iterator的子類,按常理來說應該可以執行next()方法,但是值得注意的是,KeyIterator是hashmap的內部類,JAVA給的提示是cannot access a member of class java.util.HashMap$KeyIterator withmodifiers "public"

從上面的那些例子上也能看出,除內部類外的其他類的應用上.class功能完全等於.getClass()!只是一個是用類直接獲得的,一個是用實例獲得的。


 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章