1. 程式人生 > >Java基礎之—反射

Java基礎之—反射

反射是框架設計的靈魂

(使用的前提條件:必須先得到代表的位元組碼的Class,Class類用於表示.class檔案(位元組碼))

一、反射的概述

JAVA反射機制是在執行狀態中,對於任意一個類,都能夠知道這個類的所有屬性和方法;對於任意一個物件,都能夠呼叫它的任意一個方法和屬性;這種動態獲取的資訊以及動態呼叫物件的方法的功能稱為java語言的反射機制。

要想解剖一個類,必須先要獲取到該類的位元組碼檔案物件。而解剖使用的就是Class類中的方法.所以先要獲取到每一個位元組碼檔案對應的Class型別的物件.

以上的總結就是什麼是反射

反射就是把java類中的各種成分對映成一個個的Java物件

例如:一個類有:成員變數、方法、構造方法、包等等資訊,利用反射技術可以對一個類進行解剖,把個個組成部分對映成一個個物件。

     (其實:一個類中這些成員方法、構造方法、在加入類中都有一個類來描述)

如圖是類的正常載入過程:反射的原理在與class物件。

熟悉一下載入的時候:Class物件的由來是將class檔案讀入記憶體,併為之建立一個Class物件。

其中這個Class物件很特殊。我們先了解一下這個Class類

二、檢視Class類在java中的api詳解(1.7的API)

如何閱讀java中的api詳見java基礎之——String字串處理

Class

 類的例項表示正在執行的 Java 應用程式中的類和介面。也就是jvm中有N多的例項每個類都有該Class物件。(包括基本資料型別)

Class 沒有公共構造方法。Class 物件是在載入類時由 Java 虛擬機器以及通過呼叫類載入器中的defineClass方法自動構造的。也就是這不需要我們自己去處理建立,JVM已經幫我們建立好了。

沒有公共的構造方法,方法共有64個太多了。下面用到哪個就詳解哪個吧

三、反射的使用(這裡使用Student類做演示)

先寫一個Student類。

1、獲取Class物件的三種方式

1.1 Object ——> getClass(); 1.2 任何資料型別(包括基本資料型別)都有一個“靜態”的class屬性 1.3 通過Class類的靜態方法:forName(String  className)(常用)

其中1.1是因為Object類中的getClass方法、因為所有類都繼承Object類。從而呼叫Object類來獲取

  1. <span style="font-size:18px;">package fanshe;  
  2. /** 
  3.  * 獲取Class物件的三種方式 
  4.  * 1 Object ——> getClass(); 
  5.  * 2 任何資料型別(包括基本資料型別)都有一個“靜態”的class屬性 
  6.  * 3 通過Class類的靜態方法:forName(String  className)(常用) 
  7.  * 
  8.  */  
  9. public class Fanshe {  
  10.     public static void main(String[] args) {  
  11.         //第一種方式獲取Class物件    
  12.         Student stu1 = new Student();//這一new 產生一個Student物件,一個Class物件。  
  13.         Class stuClass = stu1.getClass();//獲取Class物件  
  14.         System.out.println(stuClass.getName());  
  15.         //第二種方式獲取Class物件  
  16.         Class stuClass2 = Student.class;  
  17.         System.out.println(stuClass == stuClass2);//判斷第一種方式獲取的Class物件和第二種方式獲取的是否是同一個  
  18.         //第三種方式獲取Class物件  
  19.         try {  
  20.             Class stuClass3 = Class.forName("fanshe.Student");//注意此字串必須是真實路徑,就是帶包名的類路徑,包名.類名  
  21.             System.out.println(stuClass3 == stuClass2);//判斷三種方式是否獲取的是同一個Class物件  
  22.         } catch (ClassNotFoundException e) {  
  23.             e.printStackTrace();  
  24.         }  
  25.     }  
  26. }</span>  

注意:在執行期間,一個類,只有一個Class物件產生。

三種方式常用第三種,第一種物件都有了還要反射干什麼。第二種需要匯入類的包,依賴太強,不導包就拋編譯錯誤。一般都第三種,一個字串可以傳入也可寫在配置檔案中等多種方法。

2、通過反射獲取構造方法並使用:

student類:

  1. package fanshe;  
  2. public class Student {  
  3.     //---------------構造方法-------------------  
  4.     //(預設的構造方法)  
  5.     Student(String str){  
  6.         System.out.println("(預設)的構造方法 s = " + str);  
  7.     }  
  8.     //無參構造方法  
  9.     public Student(){  
  10.         System.out.println("呼叫了公有、無參構造方法執行了。。。");  
  11.     }  
  12.     //有一個引數的構造方法  
  13.     public Student(char name){  
  14.         System.out.println("姓名:" + name);  
  15.     }  
  16.     //有多個引數的構造方法  
  17.     public Student(String name ,int age){  
  18.         System.out.println("姓名:"+name+"年齡:"+ age);//這的執行效率有問題,以後解決。  
  19.     }  
  20.     //受保護的構造方法  
  21.     protected Student(boolean n){  
  22.         System.out.println("受保護的構造方法 n = " + n);  
  23.     }  
  24.     //私有構造方法  
  25.     private Student(int age){  
  26.         System.out.println("私有的構造方法   年齡:"+ age);  
  27.     }  
  28. }  

共有6個構造方法;

測試類:

  1. package fanshe;  
  2. import java.lang.reflect.Constructor;  
  3. /* 
  4.  * 通過Class物件可以獲取某個類中的:構造方法、成員變數、成員方法;並訪問成員; 
  5.  *  
  6.  * 1.獲取構造方法: 
  7.  *      1).批量的方法: 
  8.  *          public Constructor[] getConstructors():所有"公有的"構造方法 
  9.             public Constructor[] getDeclaredConstructors():獲取所有的構造方法(包括私有、受保護、預設、公有) 
  10.  *      2).獲取單個的方法,並呼叫: 
  11.  *          public Constructor getConstructor(Class... parameterTypes):獲取單個的"公有的"構造方法: 
  12.  *          public Constructor getDeclaredConstructor(Class... parameterTypes):獲取"某個構造方法"可以是私有的,或受保護、預設、公有; 
  13.  *       
  14.  *          呼叫構造方法: 
  15.  *          Constructor-->newInstance(Object... initargs) 
  16.  */  
  17. public class Constructors {  
  18.     public static void main(String[] args) throws Exception {  
  19.         //1.載入Class物件  
  20.         Class clazz = Class.forName("fanshe.Student");  
  21.         //2.獲取所有公有構造方法  
  22.         System.out.println("**********************所有公有構造方法*********************************");  
  23.         Constructor[] conArray = clazz.getConstructors();  
  24.         for(Constructor c : conArray){  
  25.             System.out.println(c);  
  26.         }  
  27.         System.out.println("************所有的構造方法(包括:私有、受保護、預設、公有)***************");  
  28.         conArray = clazz.getDeclaredConstructors();  
  29.         for(Constructor c : conArray){  
  30.             System.out.println(c);  
  31.         }  
  32.         System.out.println("*****************獲取公有、無參的構造方法*******************************");  
  33.         Constructor con = clazz.getConstructor(null);  
  34.         //1>、因為是無參的構造方法所以型別是一個null,不寫也可以:這裡需要的是一個引數的型別,切記是型別  
  35.         //2>、返回的是描述這個無參建構函式的類物件。  
  36.         System.out.println("con = " + con);  
  37.         //呼叫構造方法  
  38.         Object obj = con.newInstance();  
  39.     //  System.out.println("obj = " + obj);  
  40.     //  Student stu = (Student)obj;  
  41.         System.out.println("******************獲取私有構造方法,並呼叫*******************************");  
  42.         con = clazz.getDeclaredConstructor(char.class);  
  43.         System.out.println(con);  
  44.         //呼叫構造方法  
  45.         con.setAccessible(true);//暴力訪問(忽略掉訪問修飾符)  
  46.         obj = con.newInstance('男');  
  47.     }  
  48. }  

後臺輸出:

  1. **********************所有公有構造方法*********************************  
  2. public fanshe.Student(java.lang.String,int)  
  3. public fanshe.Student(char)  
  4. public fanshe.Student()  
  5. ************所有的構造方法(包括:私有、受保護、預設、公有)***************  
  6. private fanshe.Student(int)  
  7. protected fanshe.Student(boolean)  
  8. public fanshe.Student(java.lang.String,int)  
  9. public fanshe.Student(char)  
  10. public fanshe.Student()  
  11. fanshe.Student(java.lang.String)  
  12. *****************獲取公有、無參的構造方法*******************************  
  13. con = public fanshe.Student()  
  14. 呼叫了公有、無參構造方法執行了。。。  
  15. ******************獲取私有構造方法,並呼叫*******************************  
  16. public fanshe.Student(char)  
  17. 姓名:男  

呼叫方法:

1.獲取構造方法:

 1).批量的方法: public Constructor[] getConstructors():所有"公有的"構造方法             public Constructor[] getDeclaredConstructors():獲取所有的構造方法(包括私有、受保護、預設、公有)         2).獲取單個的方法,並呼叫: public Constructor getConstructor(Class... parameterTypes):獲取單個的"公有的"構造方法: public Constructor getDeclaredConstructor(Class... parameterTypes):獲取"某個構造方法"可以是私有的,或受保護、預設、公有;   呼叫構造方法:

Constructor-->newInstance(Object... initargs)

2、newInstance是 Constructor類的方法(管理建構函式的類)

api的解釋為:

newInstance(Object... initargs)            使用此 Constructor 物件表示的構造方法來建立該構造方法的宣告類的新例項,並用指定的初始化引數初始化該例項。

它的返回值是T型別,所以newInstance是建立了一個構造方法的宣告類的新例項物件。併為之呼叫

3、獲取成員變數並呼叫

student類:

  1. <span style="font-size:14px;">package fanshe.field;  
  2. public class Student {  
  3.     public Student(){  
  4.     }  
  5.     //**********欄位*************//  
  6.     public String name;  
  7.     protected int age;  
  8.     char sex;  
  9.     private String phoneNum;  
  10.     @Override  
  11.     public String toString() {  
  12.         return "Student [name=" + name + ", age=" + age + ", sex=" + sex  
  13.                 + ", phoneNum=" + phoneNum + "]";  
  14.     }  
  15. }</span>  

測試類:

  1. <span style="font-size:14px;">package fanshe.field;  
  2. import java.lang.reflect.Field;  
  3. /* 
  4.  * 獲取成員變數並呼叫: 
  5.  *  
  6.  * 1.批量的 
  7.  *      1).Field[] getFields():獲取所有的"公有欄位" 
  8.  *      2).Field[] getDeclaredFields():獲取所有欄位,包括:私有、受保護、預設、公有; 
  9.  * 2.獲取單個的: 
  10.  *      1).public Field getField(String fieldName):獲取某個"公有的"欄位; 
  11.  *      2).public Field getDeclaredField(String fieldName):獲取某個欄位(可以是私有的) 
  12.  *  
  13.  *   設定欄位的值: 
  14.  *      Field --> public void set(Object obj,Object value): 
  15.  *                  引數說明: 
  16.  *                  1.obj:要設定的欄位所在的物件; 
  17.  *                  2.value:要為欄位設定的值; 
  18.  *  
  19.  */  
  20. public class Fields {  
  21.         public static void main(String[] args) throws Exception {  
  22.             //1.獲取Class物件  
  23.             Class stuClass = Class.forName("fanshe.field.Student");  
  24.             //2.獲取欄位  
  25.             System.out.println("************獲取所有公有的欄位********************");  
  26.             Field[] fieldArray = stuClass.getFields();  
  27.             for(Field f : fieldArray){  
  28.                 System.out.println(f);  
  29.             }  
  30.             System.out.println("************獲取所有的欄位(包括私有、受保護、預設的)********************");  
  31.             fieldArray = stuClass.getDeclaredFields();  
  32.             for(Field f : fieldArray){  
  33.                 System.out.println(f);  
  34.             }  
  35.             System.out.println("*************獲取公有欄位**並呼叫***********************************");  
  36.             Field f = stuClass.getField("name");  
  37.             System.out.println(f);  
  38.             //獲取一個物件  
  39.             Object obj = stuClass.getConstructor().newInstance();//產生Student物件--》Student stu = new Student();  
  40.             //為欄位設定值  
  41.             f.set(obj, "劉德華");//為Student物件中的name屬性賦值--》stu.name = "劉德華"  
  42.             //驗證  
  43.             Student stu = (Student)obj;  
  44.             System.out.println("驗證姓名:" + stu.name);  
  45.             System.out.println("**************獲取私有欄位****並呼叫********************************");  
  46.             f = stuClass.getDeclaredField("phoneNum");  
  47.             System.out.println(f);  
  48.             f.setAccessible(true);//暴力反射,解除私有限定  
  49.             f.set(obj, "18888889999");  
  50.             System.out.println("驗證電話:" + stu);  
  51.         }  
  52.     }</span><span style="font-size: 18px;">  
  53. </span>  

後臺輸出:

  1. ************獲取所有公有的欄位********************  
  2. public java.lang.String fanshe.field.Student.name  
  3. ************獲取所有的欄位(包括私有、受保護、預設的)********************  
  4. public java.lang.String fanshe.field.Student.name  
  5. protected int fanshe.field.Student.age  
  6. char fanshe.field.Student.sex  
  7. private java.lang.String fanshe.field.Student.phoneNum  
  8. *************獲取公有欄位**並呼叫***********************************  
  9. public java.lang.String fanshe.field.Student.name  
  10. 驗證姓名:劉德華  
  11. **************獲取私有欄位****並呼叫********************************  
  12. private java.lang.String fanshe.field.Student.phoneNum  
  13. 驗證電話:Student [name=劉德華, age=0, sex=  

由此可見

呼叫欄位時:需要傳遞兩個引數:

Object obj = stuClass.getConstructor().newInstance();//產生Student物件--》Student stu = new Student(); //為欄位設定值 f.set(obj, "劉德華");//為Student物件中的name屬性賦值--》stu.name = "劉德華"

第一個引數:要傳入設定的物件,第二個引數:要傳入實參

4、獲取成員方法並呼叫

student類:

  1. <span style="font-size:14px;">package fanshe.method;  
  2. public class Student {  
  3.     //**************成員方法***************//  
  4.     public void show1(String s){  
  5.         System.out.println("呼叫了:公有的,String引數的show1(): s = " + s);  
  6.     }  
  7.     protected void show2(){  
  8.         System.out.println("呼叫了:受保護的,無參的show2()");  
  9.     }  
  10.     void show3(){  
  11.         System.out.println("呼叫了:預設的,無參的show3()");  
  12.     }  
  13.     private String show4(int age){  
  14.         System.out.println("呼叫了,私有的,並且有返回值的,int引數的show4(): age = " + age);  
  15.         return "abcd";  
  16.     }  
  17. }  
  18. </span>  

測試類:

  1. <span style="font-size:14px;">package fanshe.method;  
  2. import java.lang.reflect.Method;  
  3. /* 
  4.  * 獲取成員方法並呼叫: 
  5.  *  
  6.  * 1.批量的: 
  7.  *      public Method[] getMethods():獲取所有"公有方法";(包含了父類的方法也包含Object類) 
  8.  *      public Method[] getDeclaredMethods():獲取所有的成員方法,包括私有的(不包括繼承的) 
  9.  * 2.獲取單個的: 
  10.  *      public Method getMethod(String name,Class<?>... parameterTypes): 
  11.  *                  引數: 
  12.  *                      name : 方法名; 
  13.  *                      Class ... : 形參的Class型別物件 
  14.  *      public Method getDeclaredMethod(String name,Class<?>... parameterTypes) 
  15.  *  
  16.  *   呼叫方法: 
  17.  *      Method --> public Object invoke(Object obj,Object... args): 
  18.  *                  引數說明: 
  19.  *                  obj : 要呼叫方法的物件; 
  20.  *                  args:呼叫方式時所傳遞的實參; 
  21. ): 
  22.  */  
  23. public class MethodClass {  
  24.     public static void main(String[] args) throws Exception {  
  25.         //1.獲取Class物件  
  26.         Class stuClass = Class.forName("fanshe.method.Student");  
  27.         //2.獲取所有公有方法  
  28.         System.out.println("***************獲取所有的”公有“方法*******************");  
  29.         stuClass.getMethods();  
  30.         Method[] methodArray = stuClass.getMethods();  
  31.         for(Method m : methodArray){  
  32.             System.out.println(m);  
  33.         }  
  34.         System.out.println("***************獲取所有的方法,包括私有的*******************");  
  35.         methodArray = stuClass.getDeclaredMethods();  
  36.         for(Method m : methodArray){  
  37.             System.out.println(m);  
  38.         }  
  39.         System.out.println("***************獲取公有的show1()方法*******************");  
  40.         Method m = stuClass.getMethod("show1", String.class);  
  41.         System.out.println(m);  
  42.         //例項化一個Student物件  
  43.         Object obj = stuClass.getConstructor().newInstance();  
  44.         m.invoke(obj, "劉德華");  
  45.         System.out.println("***************獲取私有的show4()方法******************");  
  46.         m = stuClass.getDeclaredMethod("show4"int.class);  
  47.         System.out.println(m);  
  48.         m.setAccessible(true);//解除私有限定  
  49.         Object result = m.invoke(obj, 20);//需要兩個引數,一個是要呼叫的物件(獲取有反射),一個是實參  
  50.         System.out.println("返回值:" + result);  
  51.     }  
  52. }  
  53. </span>  

控制檯輸出:

  1. ***************獲取所有的”公有“方法*******************  
  2. public void fanshe.method.Student.show1(java.lang.String)  
  3. public final void java.lang.Object.wait(long,intthrows java.lang.InterruptedException  
  4. public final native void java.lang.Object.wait(longthrows java.lang.InterruptedException  
  5. public final void java.lang.Object.wait() throws java.lang.InterruptedException  
  6. public boolean java.lang.Object.equals(java.lang.Object)  
  7. public java.lang.String java.lang.Object.toString()  
  8. public native int java.lang.Object.hashCode()  
  9. public final native java.lang.Class java.lang.Object.getClass()  
  10. public final native void java.lang.Object.notify()  
  11. public final native void java.lang.Object.notifyAll()  
  12. ***************獲取所有的方法,包括私有的*******************  
  13. public void fanshe.method.Student.show1(java.lang.String)  
  14. private java.lang.String fanshe.method.Student.show4(int)  
  15. protected void fanshe.method.Student.show2()  
  16. void fanshe.method.Student.show3()  
  17. ***************獲取公有的show1()方法*******************  
  18. public void fanshe.method.Student.show1(java.lang.String)  
  19. 呼叫了:公有的,String引數的show1(): s = 劉德華  
  20. ***************獲取私有的show4()方法******************  
  21. private java.lang.String fanshe.method.Student.show4(int)  
  22. 呼叫了,私有的,並且有返回值的,int引數的show4(): age = 20  
  23. 返回值:abcd  

由此可見:

m = stuClass.getDeclaredMethod("show4", int.class);//呼叫制定方法(所有包括私有的),需要傳入兩個引數,第一個是呼叫的方法名稱,第二個是方法的形參型別,切記是型別。 System.out.println(m); m.setAccessible(true);//解除私有限定 Object result = m.invoke(obj, 20);//需要兩個引數,一個是要呼叫的物件(獲取有反射),一個是實參 System.out.println("返回值:" + result);//

控制檯輸出:

  1. ***************獲取所有的”公有“方法*******************  
  2. public void fanshe.method.Student.show1(java.lang.String)  
  3. public final void java.lang.Object.wait(long,intthrows java.lang.InterruptedException  
  4. public final native void java.lang.Object.wait(longthrows java.lang.InterruptedException  
  5. public final void java.lang.Object.wait() throws java.lang.InterruptedException  
  6. public boolean java.lang.Object.equals(java.lang.Object)  
  7. public java.lang.String java.lang.Object.toString()  
  8. public native int java.lang.Object.hashCode()  
  9. public final native java.lang.Class java.lang.Object.getClass()  
  10. public final native void java.lang.Object.notify()  
  11. public final native void java.lang.Object.notifyAll()  
  12. ***************獲取所有的方法,包括私有的*******************  
  13. public void fanshe.method.Student.show1(java.lang.String)  
  14. private java.lang.String fanshe.method.Student.show4(int)  
  15. protected void fanshe.method.Student.show2()  
  16. void fanshe.method.Student.show3()  
  17. ***************獲取公有的show1()方法*******************  
  18. public void fanshe.method.Student.show1(java.lang.String)  
  19. 呼叫了:公有的,String引數的show1(): s = 劉德華  
  20. ***************獲取私有的show4()方法******************  
  21. private java.lang.String fanshe.method.Student.show4(int)  
  22. 呼叫了,私有的,並且有返回值的,int引數的show4(): age = 20  
  23. 返回值:abcd  

其實這裡的成員方法:在模型中有屬性一詞,就是那些setter()方法和getter()方法。還有欄位組成,這些內容在內省中詳解

5、反射main方法

student類:

  1. <span style="font-size:14px;">package fanshe.main;  
  2. public class Student {  
  3.     public static void main(String[] args) {  
  4.         System.out.println("main方法執行了。。。");  
  5.     }  
  6. }  
  7. </span>  

測試類:

  1. <span style="font-size:14px;">package fanshe.main;  
  2. import java.lang.reflect.Method;  
  3. /** 
  4.  * 獲取Student類的main方法、不要與當前的main方法搞混了 
  5.  */  
  6. public class Main {  
  7.     public static void main(String[] args) {  
  8.         try {  
  9.             //1、獲取Student物件的位元組碼  
  10.             Class clazz = Class.forName("fanshe.main.Student");  
  11.             //2、獲取main方法  
  12.              Method methodMain = clazz.getMethod("main", String[].class);//第一個引數:方法名稱,第二個引數:方法形參的型別,  
  13.             //3、呼叫main方法  
  14.             // methodMain.invoke(null, new String[]{"a","b","c"});  
  15.              //第一個引數,物件型別,因為方法是static靜態的,所以為null可以,第二個引數是String陣列,這裡要注意在jdk1.4時是陣列,jdk1.5之後是可變引數  
  16.              //這裡拆的時候將  new String[]{"a","b","c"} 拆成3個物件。。。所以需要將它強轉。  
  17.              methodMain.invoke(null, (Object)new String[]{"a","b","c"});//方式一  
  18.             // methodMain.invoke(null, new Object[]{new String[]{"a","b","c"}});//方式二  
  19.         } catch (Exception e) {  
  20.             e.printStackTrace();  
  21.         }  
  22.     }  
  23. }</span><span style="font-size:18px;">  
  24. </span>  

控制檯輸出:

main方法執行了。。。

6、反射方法的其它使用之---通過反射執行配置檔案內容

student類:

  1. public class Student {  
  2.     public void show(){  
  3.         System.out.println("is show()");  
  4.     }  
  5. }  

配置檔案以txt檔案為例子(pro.txt):

  1. className = cn.fanshe.Student  
  2. methodName = show  

測試類:

  1. import java.io.FileNotFoundException;  
  2. import java.io.FileReader;  
  3. import java.io.IOException;  
  4. import java.lang.reflect.Method;  
  5. import java.util.Properties;  
  6. /* 
  7.  * 我們利用反射和配置檔案,可以使:應用程式更新時,對原始碼無需進行任何修改 
  8.  * 我們只需要將新類傳送給客戶端,並修改配置檔案即可 
  9.  */  
  10. public class Demo {  
  11.     public static void main(String[] args) throws Exception {