java中Class對象詳解和類名.class, class.forName(), getClass()區別

一直在想.class和.getClass()的區別,思索良久,有點思緒,然後有網上搜了搜,找到了如下的一篇文章,與大家分享。

 

原來爲就是涉及到Java的反射~~~~~

Java反射學習

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

1,獲得類型類

我們知道在Java中一切都是對象,我們一般所使用的對象都直接或間接繼承自Object類。Object類中包含一個方法名叫getClass,利用這個方法就可以獲得一個實例的類型類。類型類指的是代表一個類型的類,因爲一切皆是對象,類型也不例外,在Java使用類型類來表示一個類型。所有的類型類都是Class類的實例。例如,有如下一段代碼
  1. A a = new A();  
  2.   
  3. if(a.getClass()==A.class) {  
  4.   
  5.       System.out.println("equal");  
  6.   
  7. else {  
  8.   
  9.       System.out.println("unequal");  
  10.   
  11. }  
  12. 輸出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.   
  3. if(a.getClass()==B.class) {  
  4.   
  5.         System.out.println("equal");  
  6.   
  7. }  else {  
  8.           
  9.           System.out.println("unequal");  
  10. }  
  11. 輸出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:將這個類型

-----------------------------------------------------------------------------------

在學習反射時想到了這個問題,.getClass()和.class有沒有什麼區別?
       當然,最明顯的區別就是.getClass()是一個對象實例的方法,只有對象實例纔有這個方法,具體的類是沒有的。類的Class類實例是通過.class獲得的,顯然,類沒有.getClass()方法。
       從網上還找到一些其他資料:
       1、出現的時期不同:Class.forName()在運行時加載;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()!只是一個是用類直接獲得的,一個是用實例獲得的。

-------------------------------------------------------------------------------------------------------------------------------------------

Class對象的生成方式如下:

 

1.類名.class           說明: JVM將使用類裝載器, 將類裝入內存(前提是:類還沒有裝入內存),不做類的初始化工作.返回Class的對象

 

2.Class.forName("類名字符串")  (注:類名字符串是包名+類名)  說明:裝入類,並做類的靜態初始化,返回Class的對象

 

3.實例對象.getClass()  說明:對類進行靜態初始化、非靜態初始化;返回引用運行時真正所指的對象(因爲:子對象的引用可能會賦給父對象的引用變量中)所屬的類的Class的對象

 

通過下面的程序,來觀察一下Class對象的生成的原理。

 

[java] view plain copy 在CODE上查看代碼片派生到我的代碼片
  1. package ClassTest;  
  2.   
  3. public class TestClass {  
  4.     public static void main(String[] args) {  
  5.   
  6.         try {  
  7.             // 測試.class  
  8.             @SuppressWarnings("rawtypes")  
  9.             Class testTypeClass = TestClassType.class;  
  10.             System.out.println("testTypeClass---" + testTypeClass);  
  11.   
  12.             // 測試Class.forName()  
  13.             @SuppressWarnings("rawtypes")  
  14.             Class testTypeForName = Class.forName("ClassTest.TestClassType");  
  15.             System.out.println("testTypeForName---" + testTypeForName);  
  16.   
  17.             // 測試Object.getClass()  
  18.             TestClassType testTypeGetClass = new TestClassType();  
  19.             System.out.println("testTypeGetClass---"  
  20.                     + testTypeGetClass.getClass());  
  21.   
  22.         } catch (ClassNotFoundException e) {  
  23.             // TODO Auto-generated catch block  
  24.             e.printStackTrace();  
  25.         }  
  26.     }  
  27.   
  28. }  

[java] view plain copy 在CODE上查看代碼片派生到我的代碼片
  1. package ClassTest;  
  2.   
  3. public class  TestClassType {  
  4.   
  5.     // 構造函數  
  6.     public TestClassType() {  
  7.         System.out.println("----構造函數---");  
  8.     }  
  9.   
  10.     // 靜態的參數初始化  
  11.     static {  
  12.         System.out.println("---靜態的參數初始化---");  
  13.     }  
  14.   
  15.     // 非靜態的參數初始化  
  16.     {  
  17.         System.out.println("----非靜態的參數初始化---");  
  18.     }  
  19.   
  20. }  


 運行結果如下

[java] view plain copy 在CODE上查看代碼片派生到我的代碼片
  1. testTypeClass---class ClassTest.TestClassType  
  2. ---靜態的參數初始化---  
  3. testTypeForName---class ClassTest.TestClassType  
  4. ----非靜態的參數初始化---  
  5. ----構造函數---  
  6. testTypeGetClass---class ClassTest.TestClassType  

 然而單獨執行  測試Class.forName() 時結果爲

---靜態的參數初始化e---
testTypeForName---測試Class.forName()/class ClassTest.TestClassType

 

單獨執行測試Object.getClass() 結果爲 

---靜態的參數初始化e---
----非靜態的參數初始化e---
----構造函數e---
testTypeGetClass---測試Object.getClass()/class ClassTest.TestClassType

 

所以根據結果可以發現,三種生成的Class對象一樣的。並且程序只打印一次“靜態的參數初始化”,但時拆分單獨執行時class.forName(),Object.getClass()時

都執行了“靜態的參數初始化”。

我們知道,靜態的方法屬性初始化,是在加載類的時候初始化。而非靜態方法屬性初始化,是new 類實例對象的時候加載。

因此,這段程序說明,三種方式生成Class對象,其實只有一個Class對象。在生成Class對象的時候,首先判斷內存中是否已經加載。

所以,生成Class對象的過程其實是如此的:

當我們編寫一個新的java類時,JVM就會幫我們編譯成class對象,存放在同名的.class文件中。在運行時,當需要生成這個類的對象,JVM就會檢查此類是否已經裝載內存中。若是沒有裝載,則把.class文件裝入到內存中。若是裝載,則根據class文件生成實例對象。

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