Spring 在創(chuàng)建 Bean 實(shí)例和依賴注入以及AOP時(shí)都使用了反射,今天我們就來講解一下反射的概念以及其應(yīng)用。
反射機(jī)制
Java反射是Java被視為動(dòng)態(tài)(或準(zhǔn)動(dòng)態(tài))語言的一個(gè)關(guān)鍵性質(zhì)。這個(gè)機(jī)制允許程序在運(yùn)行時(shí)通過Reflection APIs取得任何一個(gè)已知名稱的class的內(nèi)部信息以及任意一個(gè)對(duì)象的內(nèi)部信息。Java反射機(jī)制提供如下功能:
在運(yùn)行時(shí)判斷任意一個(gè)對(duì)象所屬的類
在運(yùn)行時(shí)構(gòu)造任意一個(gè)類的對(duì)象
在運(yùn)行時(shí)判斷任意一個(gè)類所具有的成員變量和方法
在運(yùn)行時(shí)調(diào)用任一個(gè)對(duì)象的方法
在運(yùn)行時(shí)創(chuàng)建新類對(duì)象
在使用Java的反射功能時(shí),基本首先都要獲取類的Class對(duì)象,再通過Class對(duì)象獲取其他的對(duì)象。大家都知道,在Junit4中注解@Test表示測(cè)試用例,每一個(gè)測(cè)試用例的本質(zhì)就是測(cè)試類中的一個(gè)方法,即:
- @Test
- public void test() {
- fail("Not yet implemented");
- }
我們知道,通常情況下,調(diào)用一個(gè)類的方法是,先對(duì)類進(jìn)行實(shí)例化,記為obj,然后通過obj.test()的方式調(diào)用。在這里我們思考一個(gè)問題,Junit4是一個(gè)框架,在運(yùn)行的過程中,框架根本不知道用戶定義了多少個(gè)測(cè)試用例(雖然通過@Test進(jìn)行了約束),顯然框架是在運(yùn)行的時(shí)候才確認(rèn)了測(cè)試用例,并通過某種方式調(diào)用了測(cè)試用例,這就是反射的本質(zhì)——在運(yùn)行時(shí)工作!
Class類和Class實(shí)例
我們知道Java中的類是一個(gè)模板,它描述一類對(duì)象的行為和狀態(tài),例如:
- class Person{
- }
- Person kevin=new Person();
- Person mike=new Person();
Person就是Kevin、Mike這兩個(gè)對(duì)象的類型,即是Kevin和Mike兩個(gè)對(duì)象的描述。
Java中一切皆對(duì)象,那么Person(自定義類)、String(JDK提供的類)...又是什么類型呢?他們都是Class類的對(duì)象,都由Class類來描述。
Class的實(shí)例是什么?是類或接口,更嚴(yán)格地說是java中的字節(jié)碼(類或接口編譯后生成的.class文件)。
常用API介紹
在這里我們重點(diǎn)介紹反射技術(shù)中關(guān)于獲取Class對(duì)象,訪問字段,調(diào)用方法以及調(diào)用構(gòu)造方法的API
1.獲取類的Class對(duì)象
Class(java.lang.Class) 類的實(shí)例表示正在運(yùn)行的 Java 應(yīng)用程序中的類和接口。這個(gè)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為每個(gè)加載的class創(chuàng)建了對(duì)應(yīng)的Class實(shí)例,并在實(shí)例中保存了該class的所有信息,包括類名、包名、父類、實(shí)現(xiàn)的接口、所有方法、字段等,因此,如果獲取了某個(gè)Class實(shí)例,我們就可以通過這個(gè)Class實(shí)例獲取到該實(shí)例對(duì)應(yīng)的class的所有信息。獲取類的Class對(duì)象有多種方式:
2、獲取類的Fields
可以通過反射機(jī)制得到某個(gè)類的某個(gè)屬性,然后改變對(duì)應(yīng)于這個(gè)類的某個(gè)實(shí)例的該屬性值。JAVA 的Class
3.獲取類的Method
通過反射機(jī)制得到某個(gè)類的某個(gè)方法,然后調(diào)用對(duì)應(yīng)于這個(gè)類的某個(gè)實(shí)例的該方法,Class
4.獲取類的Constructor
通過反射機(jī)制得到某個(gè)類的構(gòu)造器,然后調(diào)用該構(gòu)造器創(chuàng)建該類的一個(gè)實(shí)例,Class
反射API應(yīng)用
寫一個(gè)類
- 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對(duì)象的newInstance方法,該方法會(huì)調(diào)用對(duì)象的默認(rèn)構(gòu)造器,如果沒有默認(rèn)構(gòu)造器,會(huì)調(diào)用失敗,代碼如下:
- Class classType =ReflectDemo.class;
- Object inst = classType.newInstance();
- System.out.println(inst);
調(diào)用默認(rèn)Constructor對(duì)象的newInstance方法,代碼如下:
- Class classType =ReflectDemo.class;
- Constructor constructor1 = classType.getConstructor();
- Object inst = constructor1.newInstance();
- System.out.println(inst);
調(diào)用帶參數(shù)Constructor對(duì)象的newInstance方法,代碼如下:
- Constructor constructor2 =ReflectDemo.class.getDeclaredConstructor(String.class);
- Object inst = constructor2.newInstance("test");
- System.out.println(inst);
調(diào)用方法
通過反射獲取類Method對(duì)象,獲取類中的所有函數(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對(duì)象,調(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對(duì)象,調(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à)值.
原文地址:https://www.toutiao.com/i6947870602661200388/