原來爲就是涉及到Java的反射~~~~~
Java反射學習
所謂反射,可以理解爲在運行時期獲取對象類型信息的操作。傳統的編程方法要求程序員在編譯階段決定使用的類型,但是在反射的幫助下,編程人員可以動態獲取這些信息,從而編寫更加具有可移植性的代碼。嚴格地說,反射並非編程語言的特性,因爲在任何一種語言都可以實現反射機制,但是如果編程語言本身支持反射,那麼反射的實現就會方便很多。
1,獲得類型類
我們知道在Java中一切都是對象,我們一般所使用的對象都直接或間接繼承自Object類。Object類中包含一個方法名叫getClass,利用這個方法就可以獲得一個實例的類型類。類型類指的是代表一個類型的類,因爲一切皆是對象,類型也不例外,在Java使用類型類來表示一個類型。所有的類型類都是Class類的實例。例如,有如下一段代碼- A a = new A();
- if(a.getClass()==A.class) {
- System.out.println("equal");
- } else {
- System.out.println("unequal");
- }
- 輸出equal;
可以看到,對象a是A的一個實例,A是某一個類,在if語句中使用a.getClass()返回的結果正是類A的類型類,在Java中表示一個特定類型的類型類可以用“類型.class”的方式獲得,因爲a.getClass()獲得是A的類型類,也就是A.class,因此上面的代碼執行的結果就是打印出“equal”。特別注意的是,類型類是一一對應的,父類的類型類和子類的類型類是不同的,因此,假設A是B的子類,那麼如下的代碼將得到“unequal”的輸出:
- A a = new A();
- if(a.getClass()==B.class) {
- System.out.println("equal");
- } else {
- System.out.println("unequal");
- }
- 輸出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對象的生成的原理。
- package ClassTest;
- public class TestClass {
- public static void main(String[] args) {
- try {
- // 測試.class
- @SuppressWarnings("rawtypes")
- Class testTypeClass = TestClassType.class;
- System.out.println("testTypeClass---" + testTypeClass);
- // 測試Class.forName()
- @SuppressWarnings("rawtypes")
- Class testTypeForName = Class.forName("ClassTest.TestClassType");
- System.out.println("testTypeForName---" + testTypeForName);
- // 測試Object.getClass()
- TestClassType testTypeGetClass = new TestClassType();
- System.out.println("testTypeGetClass---"
- + testTypeGetClass.getClass());
- } catch (ClassNotFoundException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- package ClassTest;
- public class TestClassType {
- // 構造函數
- public TestClassType() {
- System.out.println("----構造函數---");
- }
- // 靜態的參數初始化
- static {
- System.out.println("---靜態的參數初始化---");
- }
- // 非靜態的參數初始化
- {
- System.out.println("----非靜態的參數初始化---");
- }
- }
運行結果如下
- testTypeClass---class ClassTest.TestClassType
- ---靜態的參數初始化---
- testTypeForName---class ClassTest.TestClassType
- ----非靜態的參數初始化---
- ----構造函數---
- 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文件生成實例對象。