測試開發(fā)必須掌握的知識點(diǎn):Spring的核心知識點(diǎn) -Java反射

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)建。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容