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");
因此,如果你知道一個實例,那么你可以通過實例的“getClass()”方法獲得該對象的類型類,如果你知道一個類型,那么你可以使用“.class”的方法獲得該類型的類型類。
2,獲得類型的信息
在獲得類型類之后,你就可以調用其中的一些方法獲得類型的信息了,主要的方法有:
getName():String:獲得該類型的全稱名稱。
getSuperClass():Class:獲得該類型的直接父類,如果該類型沒有直接父類,那么返回null。
getInterfaces():Class[]:獲得該類型實現的所有接口。
isArray():boolean:判斷該類型是否是數組。
isEnum():boolean:判斷該類型是否是枚舉類型。
isInterface():boolean:判斷該類型是否是接口。
isPrimitive():boolean:判斷該類型是否是基本類型,即是否是int,boolean,double等等。
isAssignableFrom(Class cls):boolean:判斷這個類型是否是類型cls的父(祖先)類或父(祖先)接口。
getComponentType():Class:如果該類型是一個數組,那么返回該數組的組件類型。
此外還可以進行類型轉換這類的操作,主要方法有:
asSubclass(Class clazz):Class:將這個類型
1 Obejct類有一個getClass()方法: 2 返回此 Object 的運行時類。 3 返回的 Class 對象是由所表示類的 static synchronized 方法鎖定的對象。 4 5 public class Art { 6 Art() { 7 System.out.println("Art"); 8 System.out.println(getClass().getName()); 9 } 10 } 11 12 public class Drawing extends Art { 13 Drawing() { 14 System.out.println("Drawing"); 15 System.out.println(getClass().getName()); 16 } 17 } 18 19 public class Cartoon extends Drawing{ 20 Cartoon(){ 21 System.out.println("Cartoon"); 22 System.out.println(getClass().getName()); 23 } 24 25 public static void main(String[] args) { 26 Cartoon x = new Cartoon(); 27 28 Drawing one = (Drawing)x; 29 Art two = (Art)x; 30 if(one == two){ 31 System.out.println("=="); 32 }else { 33 System.out.println("!="); 34 } 35 System.out.println(x.toString()); 36 System.out.println(one.toString()); 37 System.out.println(two.toString()); 38 } 39 } 40 //輸出 41 Art 42 com.cignacmc.knowledge.inheritance.Cartoon 43 Drawing 44 com.cignacmc.knowledge.inheritance.Cartoon 45 Cartoon 46 com.cignacmc.knowledge.inheritance.Cartoon 47 == 48 com.cignacmc.knowledge.inheritance.Cartoon@182f0db 49 com.cignacmc.knowledge.inheritance.Cartoon@182f0db 50 com.cignacmc.knowledge.inheritance.Cartoon@182f0db 51 52 結論: 當調用getClass()時,返回這個對象真實的Class對象。 53 從3個繼承對象相等的情況和輸出可知,這三個對象有相同的this指針,即內存地址一致。 54 而getClass()返回的就是this指針所代表的最真實的Class的對象,也即最上層的子類。