Spring 在創(chuàng)建 Bean 實(shí)例和依賴注入以及AOP時都使用了反射,今天我們就來講解一下反射的概念以及其應(yīng)用。

反射機(jī)制
Java反射是Java被視為動態(tài)(或準(zhǔn)動態(tài))語言的一個關(guān)鍵性質(zhì)。這個機(jī)制允許程序在運(yùn)行時通過Reflection APIs取得任何一個已知名稱的class的內(nèi)部信息以及任意一個對象的內(nèi)部信息。Java反射機(jī)制提供如下功能:
在運(yùn)行時判斷任意一個對象所屬的類
在運(yùn)行時構(gòu)造任意一個類的對象
在運(yùn)行時判斷任意一個類所具有的成員變量和方法
在運(yùn)行時調(diào)用任一個對象的方法
在運(yùn)行時創(chuàng)建新類對象
在使用Java的反射功能時,基本首先都要獲取類的Class對象,再通過Class對象獲取其他的對象。大家都知道,在Junit4中注解@Test表示測試用例,每一個測試用例的本質(zhì)就是測試類中的一個方法,即:
@Test
??? public void test() {
??????? fail("Not yet implemented");
??? }
我們知道,通常情況下,調(diào)用一個類的方法是,先對類進(jìn)行實(shí)例化,記為obj,然后通過obj.test()的方式調(diào)用。在這里我們思考一個問題,Junit4是一個框架,在運(yùn)行的過程中,框架根本不知道用戶定義了多少個測試用例(雖然通過@Test進(jìn)行了約束),顯然框架是在運(yùn)行的時候才確認(rèn)了測試用例,并通過某種方式調(diào)用了測試用例,這就是反射的本質(zhì)——在運(yùn)行時工作!
Class類和Class實(shí)例
java.lang.Class(大寫的Class)的作用是運(yùn)行時提供或獲得某個對象的類型信息,可用于反射。
java區(qū)分大小寫,因此Class和關(guān)鍵字class并不沖突。
當(dāng)我們創(chuàng)建一個類Dog時,Java會自動生成一個內(nèi)容是Dog的Class類的對象(虛擬機(jī)為每種類型管理一個獨(dú)一無二的Class對象)。
Class類的對象只能由JVM創(chuàng)建,無法通過new來創(chuàng)建。
?常用API介紹
在這里我們重點(diǎn)介紹反射技術(shù)中關(guān)于獲取Class對象,訪問字段,調(diào)用方法以及調(diào)用構(gòu)造方法的API
1.獲取類的Class對象
Class(java.lang.Class) 類的實(shí)例表示正在運(yùn)行的 Java 應(yīng)用程序中的類和接口。這個Class實(shí)例是JVM內(nèi)部創(chuàng)建的,如果我們查看JDK源碼,可以發(fā)現(xiàn)Class類的構(gòu)造方法是private,只有JVM能創(chuàng)建Class實(shí)例,我們自己的Java程序是無法創(chuàng)建Class實(shí)例的。由于JVM為每個加載的class創(chuàng)建了對應(yīng)的Class實(shí)例,并在實(shí)例中保存了該class的所有信息,包括類名、包名、父類、實(shí)現(xiàn)的接口、所有方法、字段等,因此,如果獲取了某個Class實(shí)例,我們就可以通過這個Class實(shí)例獲取到該實(shí)例對應(yīng)的class的所有信息。獲取類的Class對象有多種方式:

2、獲取類的Fields
可以通過反射機(jī)制得到某個類的某個屬性,然后改變對應(yīng)于這個類的某個實(shí)例的該屬性值。JAVA 的Class<T>類提供了幾個方法獲取類的屬性。

3.獲取類的Method
通過反射機(jī)制得到某個類的某個方法,然后調(diào)用對應(yīng)于這個類的某個實(shí)例的該方法,Class<T>類提供了幾個方法獲取類的方法。

4.獲取類的Constructor
通過反射機(jī)制得到某個類的構(gòu)造器,然后調(diào)用該構(gòu)造器創(chuàng)建該類的一個實(shí)例,Class<T>類提供了幾個方法獲取類的構(gòu)造器。

反射API應(yīng)用
寫一個類
public class ReflectDemo {
?????? ReflectDemo(){????
????????????? System.out.println("默認(rèn)構(gòu)造函數(shù)");
?????? }
?????? ReflectDemo(String p_para){??????
????????????? System.out.println("有參構(gòu)造函數(shù)");
?????? }
?????? public String myPara1="public屬性";
?????? protected String myPara2="protected屬性";
?????? private String myPara3="private屬性";
?????? public void test1(){
????????????? System.out.println("這是 public void 無參方法test1");
?????? }
?????? protected String test2(String p_test2){
????????????? System.out.println("這是 protected void 有參方法test2");
????????????? returnp_test2;
?????? }
?????? private void test3(){
????????????? System.out.println("這是 privated 無參方法test3");
?????? }
}
新建類實(shí)例
調(diào)用類的Class對象的newInstance方法,該方法會調(diào)用對象的默認(rèn)構(gòu)造器,如果沒有默認(rèn)構(gòu)造器,會調(diào)用失敗,代碼如下:
Class classType =ReflectDemo.class;
Object inst = classType.newInstance();
System.out.println(inst);
調(diào)用默認(rèn)Constructor對象的newInstance方法,代碼如下:
Class classType =ReflectDemo.class;
Constructor constructor1 = classType.getConstructor();
Object inst = constructor1.newInstance();
System.out.println(inst);
調(diào)用帶參數(shù)Constructor對象的newInstance方法,代碼如下:
Constructor constructor2 =ReflectDemo.class.getDeclaredConstructor(String.class);
Object inst = constructor2.newInstance("test");
System.out.println(inst);
調(diào)用方法
通過反射獲取類Method對象,獲取類中的所有函數(shù)。
String className = "com.lesson.reflect.ReflectDemo";???????
Class clas = Class.forName(className);
Method[] a=clas.getDeclaredMethods();
for(int i=0;i<a.length;i++){
?????? System.out.println(a[i].toString());
}
通過反射獲取類Method對象,調(diào)用method的Invoke方法調(diào)用函數(shù)。
//********調(diào)用protected有參方法? ,有參方法
Class simpleClass = Class.forName("com.lesson.reflect.ReflectDemo");
Object simpelObject = simpleClass.newInstance();
Method simpleMethod =simpleClass.getDeclaredMethod("test2",? String.class);
simpleMethod.invoke(simpelObject, "Hello,world");
//********調(diào)用private方法,有參方法
Class simpleClass2 = Class.forName("com.lesson.reflect.ReflectDemo");
Object simpelObject2 = simpleClass2.newInstance();
Method simpleMethod2 = simpleClass2.getDeclaredMethod("test3",?? String.class);
simpleMethod2.setAccessible(true);
simpleMethod2.invoke(simpelObject2, "Hello,world");
//********調(diào)用public,無參方法
Class simpleClass3 = Class.forName("com.lesson.reflect.ReflectDemo");
Object simpelObject3 =simpleClass3.newInstance();?????????????
Method simpleMethod3 =simpleClass3.getDeclaredMethod("test1");
simpleMethod3.invoke(simpelObject3);
設(shè)置讀取屬性
通過反射獲取類的Field對象,調(diào)用Field中的方法設(shè)置或獲取值
//********設(shè)置或獲取private變量? ???????? ??
ReflectDemo t =new ReflectDemo();
Class temp = t.getClass();
Field f;
f = temp.getDeclaredField("myPara3");
f.setAccessible(true);?
System.out.println(f.get(t));
f.set(t, "新的private屬性");
System.out.println(f.get(t));
好了,這就是反射的基礎(chǔ)API使用方法,可能大家還是不能夠理解其在實(shí)際工作中的應(yīng)用價(jià)值,接下來我會寫一篇文章把java的注解和反射知識點(diǎn)結(jié)合起來幫助大家更好的消化,敬請期待!
java.lang.Class(大寫的Class)的作用是運(yùn)行時提供或獲得某個對象的類型信息,可用于反射。
java區(qū)分大小寫,因此Class和關(guān)鍵字class并不沖突。
當(dāng)我們創(chuàng)建一個類Dog時,Java會自動生成一個內(nèi)容是Dog的Class類的對象(虛擬機(jī)為每種類型管理一個獨(dú)一無二的Class對象)。
Class類的對象只能由JVM創(chuàng)建,無法通過new來創(chuàng)建。