Java反射综述
Java讓我們在運行時識別對象和類的信息,主要有2種方式:
- 傳統的RTTI,它假定我們在編譯時已經知道了所有的類型信息
- 反射機制,它允許我們在運行時發現和使用類的信息。
Class對象
類型信息在運行時是如何表示的,這是由Class對象來完成的,它包含了與類有關的信息。Class對象就是用來創建所有“常規”對象的,Java使用Class對象來執行RTTI,即使你正在執行的是類似類型轉換這樣的操作。
每個類都會產生一個對應的Class對象,也就是保存在.class文件。所有類都是在對其第一次使用時,動態加載到JVM的,當程序創建一個對類的靜態成員的引用時,就會加載這個類。Class對象僅在需要的時候才會加載,static初始化是在類加載時進行的。
獲取Class對象的方法
//1、通過對象調用 getClass() 方法來獲取,通常應用在:比如你傳過來一個 Object
//? 類型的對象,而我不知道你具體是什么類,用這種方法
Person p1 = new Person();
Class c1 = p1.getClass();
?????? ?
//2、直接通過 類名.class 的方式得到,該方法最為安全可靠,程序性能更高
//? 這說明任何一個類都有一個隱含的靜態成員變量 class
Class c2 = Person.class;
?????? ?
//3、通過 Class 對象的 forName() 靜態方法來獲取,用的最多,
//?? 但可能拋出 ClassNotFoundException 異常
Class c3 = Class.forName("com.ys.reflex.Person")
反射
通過反射可以獲取運行時的信息,包括:
- Class對象
- 類名
- 修飾符
- 包信息
- 父類
- 實現的接口
- 構造器
- 方法
- 變量
- 注解
反射機制的常用相關類
與Java反射相關的類如下:
| Class類 | 代表類的實體,在運行的Java應用程序中表示類和接口 |
| Field類 | 代表類的成員變量(成員變量也稱為類的屬性) |
| Method類 | 代表類的方法 |
| Constructor類 | 代表類的構造方法 |
反射繼承關系圖
見:https://blog.csdn.net/demon7552003/article/details/86654082
類型檢查
-
instanceof
obj instanceof class
如果class obj1 = obj成立的話,返回true,否則返回false
instanceof運算符?只被用于對象引用變量,檢查左邊的被測試對象?是不是?右邊類或接口的?實例化。如果被測對象是null值,則測試結果總是false
-
Class.isInstance()
Class類的isInstance(Object?obj)方法,obj是被測試的對象,如果obj是調用這個方法的class或接口?的實例,則返回true。這個方法是instanceof運算符的動態等價
自身類.class.isInstance(自身實例或子類實例)??返回true
// B是A的子類,C是B的子類 B b = new B(); // true System.out.println(A.class.isInstance(b)); // true System.out.println(B.class.isInstance(b)); // false System.out.println(C.class.isInstance(b));-
Class.isAssignableFrom
Class類的isAssignableFrom(Class cls)方法,如果調用這個方法的class或接口 與 參數cls表示的類或接口相同,或者是參數cls表示的類或接口的父類,則返回true。
形象地:自身類.class.isAssignableFrom(自身類或子類.class)? 返回true
Java中創建(實例化)對象的五種方式
- 用new語句創建對象,這是最常見的創建對象的方法。
- 調用對象的clone()方法。
- 運用反射手段
調用java.lang.Class或者java.lang.reflect.Constructor類的newInstance()實例方法。
Object obj = Class.forName(“java.lang.Object”).newInstance();
- 通過I/O流(包括反序列化)
如運用反序列化手段,調用java.io.ObjectInputStream對象的 readObject()方法。
- 通過工廠方法返回對象
如:String str = String.valueOf(23)
?
?
?
總結