目錄
- 【Android】注解框架(一)-- 基礎知識Java 反射
- 【Android】注解框架(二)-- 基礎知識(Java注解)& 運行時注解框架
- 【Android】注解框架(三)-- 編譯時注解逛腿,手寫B(tài)utterKnife
- 【Android】注解框架(四)-- 一行代碼注入微信支付
定義
主要是指程序可以訪問,檢測和修改它本身狀態(tài)或行為的一種能力,并能根據(jù)自身行為的狀態(tài)和結果市栗,調整或修改應用所描述行為的狀態(tài)和相關的語義沈善。
反射機制就是在運行狀態(tài)中物舒,對于任意一個類卦洽,都能夠知道這個類的所有屬性和方法江锨;對于任意一個對象吃警,都能夠調用它的任意一個方法和屬性;這種動態(tài)獲取的信息以及動態(tài)調用對象的方法的功能稱為java語言的反射機制啄育。
用一句話總結就是反射可以實現(xiàn)在運行時可以知道任意一個類的屬性和方法酌心。
作用
- 指的是可以于運行時加載,探知和使用編譯期間完全未知的類
- 程序在運行狀態(tài)中, 可以動態(tài)加載一個只有名稱的類, 對于任意一個已經(jīng)加載的類,都能夠知道這個類的所有屬性和方法; 對于任意一個對象,都能調用他的任意一個方法和屬性
- 加載完類之后, 在堆內存中會產(chǎn)生一個Class類型的對象(一個類只有一個Class對象), 這個對象包含了完整的類的結構信息,而且這個Class對象就像一面鏡子,透過這個鏡子看到類的結構
Class對象
Class是反射的基礎,是java.lang.Class
的實例對象灸撰,并且Class是所有類的類類型谒府。
private Class(ClassLoader loader) {
// Initialize final field for classLoader. The initialization value of non-null
// prevents future JIT optimizations from assuming this final field is null.
classLoader = loader;
}
通過源碼可以看到Class的構造函數(shù)是私有的,只有JVM才可以創(chuàng)建Class對象浮毯,我們不能通過new Class()
的方式獲取到Class對象完疫。
每個類被加載進入內存之后,系統(tǒng)就會為該類生成一個對應的java.lang.Class對象,通過該Class對象就可以訪問到JVM中的這個類.
假設有Person
類,我們可以通過以下的方法來獲取到Class:
- 通過Person的對象债蓝,person.getClass()
- Person.class
- Class.forName(Person的全路徑包名)
獲取到了Class之后我們就能通過Java提供的API來訪問或者修改Person類的各種屬性和方法壳鹤。
創(chuàng)建對象
我們在獲取到了Class之后可以來創(chuàng)建對象,最常見的做法:
Class personClazz = Class.forName(com.example.Person);
Object person = personClazz.newInstance();
通過Class我們只能通過無參構造函數(shù)來創(chuàng)建對象饰迹。
另外我們也可以通過Class獲取到構造函數(shù)之后通過構造函數(shù)來創(chuàng)建對象芳誓。
獲取構造函數(shù)
// 獲得該類所有的構造器,不包括其父類的構造器
public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)
// 獲得該類所以public構造器啊鸭,包括父類
public Constructor<T> getConstructor(Class<?>... parameterTypes)
我們可以通過獲取構造函數(shù)之后來創(chuàng)建對象锹淌,舉個例子:
public class Person{
private String name;
public Person(String name){
this.name = name;
}
}
public class Test{
public static void main(String[] args){
Class clazz = Class.forName("com.example.Person");
Constructor constructor = clazz.getDeclaredConstructor(String.class);
constructor.newInstance("張三");
}
}
獲取方法
// 得到該類所有的方法,不包括父類的
public Method getDeclaredMethod(String name, Class<?>... parameterTypes)
// 得到該類所有的public方法赠制,包括父類的
public Method getMethod(String name, Class<?>... parameterTypes)
// 得到該類所有的方法赂摆,不包括父類的或者
Method[] methods = c.getDeclaredMethods();
// 得到該類所有的public方法,包括父類的
Method[] methods = c.getMethods();
獲取單個方法的Class<?>... parameterTypes
參數(shù)表示的是想反射的方法的參數(shù)的類類型。
舉個例子:
public class Person{
private String name;
public void setName(String name){
this.name = name;
}
}
public class Test{
public static void main(String[] args){
Class clazz = Class.forName("com.example.Person");
Object person = clazz.newInstance();
Method method = clazz.getDeclaredMethod("setName", String.class);
method.invoke(person, "張三");
}
}
最后通過invoke方法調用方法烟号,其中第一個參數(shù)是實例對象绊谭,后面的是參數(shù)。
獲取屬性
// 獲得該類自身聲明的所有變量汪拥,不包括其父類的變量
public Field getDeclaredField(String name)
// 獲得該類自所有的public成員變量达传,包括其父類變量
public Field getField(String name)
// 返回 Field 對象的一個數(shù)組
public Field[] getDeclaredFields()
其他
// 將此對象的 accessible 標志設置為指示的布爾值。值為 true 則指示反射的對象在使用時應該取消 Java 語言訪問檢查迫筑。值為 false 則指示反射的對象應該實施 Java 語言訪問檢查宪赶。
void setAccessible(boolean flag)
// 獲取屬性或者方法或者類上的注解
<T extends Annotation> T getAnnotation(Class<T> annotationClass)
Annotation[] getAnnotations()
其他具體方法和屬性可以參考:Java api 鏈接,搜索java.lang.reflect包脯燃。