<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    飛翔的起點

    從這里出發

    導航

    <2009年1月>
    28293031123
    45678910
    11121314151617
    18192021222324
    25262728293031
    1234567

    統計

    常用鏈接

    留言簿(5)

    隨筆分類

    隨筆檔案

    文章分類

    文章檔案

    搜索

    最新評論

    閱讀排行榜

    評論排行榜

    反射機制補充

    本篇文章為在工作中使用JAVA反射的經驗總結,也可以說是一些小技巧,以后學會新的小技巧,會不斷更新。

            在開始之前,我先定義一個測試類Student,代碼如下:

    1. package chb.test.reflect;   
    2.   
    3. public class Student {   
    4.     private int age;   
    5.     private String name;   
    6.     public int getAge() {   
    7.         return age;   
    8.      }   
    9.     public void setAge(int age) {   
    10.         this.age = age;   
    11.      }   
    12.     public String getName() {   
    13.         return name;   
    14.      }   
    15.     public void setName(String name) {   
    16.         this.name = name;   
    17.      }   
    18.        
    19.     public static void hi(int age,String name){   
    20.          System.out.println("大家好,我叫"+name+",今年"+age+"歲");   
    21.      }   
    22. }<PRE></PRE>  

    一、JAVA反射的常規使用步驟

        反射調用一般分為3個步驟:

    • 得到要調用類的class
    • 得到要調用的類中的方法(Method)
    • 方法調用(invoke)

         代碼示例:

    1. Class cls = Class.forName("chb.test.reflect.Student");   
    2. Method m = cls.getDeclaredMethod("hi",new Class[]{int.class,String.class});   
    3. m.invoke(cls.newInstance(),20,"chb");<PRE></PRE>  

    二、方法調用中的參數類型

            在方法調用中,參數類型必須正確,這里需要注意的是不能使用包裝類替換基本類型,比如不能使用Integer.class代替int.class。

           如我要調用Student的setAge方法,下面的調用是正確的:

    1. Class cls = Class.forName("chb.test.reflect.Student");   
    2. Method setMethod = cls.getDeclaredMethod("setAge",int.class);   
    3. setMethod.invoke(cls.newInstance(), 15);<PRE></PRE>  

     

           而如果我們用Integer.class替代int.class就會出錯,如:

    1. Class cls = Class.forName("chb.test.reflect.Student");   
    2. Method setMethod = cls.getDeclaredMethod("setAge",Integer.class);   
    3. setMethod.invoke(cls.newInstance(), 15);<PRE></PRE>  

     

           jvm會報出如下異常:

    1. java.lang.NoSuchMethodException: chb.test.reflect.Student.setAge(java.lang.Integer)   
    2.      at java.lang.Class.getDeclaredMethod(Unknown Source)   
    3.      at chb.test.reflect.TestClass.testReflect(TestClass.java:23)<PRE></PRE>  

     

    三、static方法的反射調用

     

           static方法調用時,不必得到對象示例,如下:

    1. Class cls = Class.forName("chb.test.reflect.Student");   
    2. Method staticMethod = cls.getDeclaredMethod("hi",int.class,String.class);   
    3. staticMethod.invoke(cls,20,"chb");//這里不需要newInstance   
    4. //staticMethod.invoke(cls.newInstance(),20,"chb");<PRE></PRE>  

    四、private的成員變量賦值

        如果直接通過反射給類的private成員變量賦值,是不允許的,這時我們可以通過setAccessible方法解決。代碼示例:

    1. Class cls = Class.forName("chb.test.reflect.Student");   
    2. Object student = cls.newInstance();//得到一個實例   
    3. Field field = cls.getDeclaredField("age");   
    4. field.set(student, 10);   
    5. System.out.println(field.get(student));<PRE></PRE>  

     

         運行如上代碼,系統會報出如下異常:

    1. java.lang.IllegalAccessException: Class chb.test.reflect.TestClass can not access a member of class chb.test.reflect.Student with modifiers "private"   
    2.      at sun.reflect.Reflection.ensureMemberAccess(Unknown Source)   
    3.      at java.lang.reflect.Field.doSecurityCheck(Unknown Source)   
    4.      at java.lang.reflect.Field.getFieldAccessor(Unknown Source)   
    5.      at java.lang.reflect.Field.set(Unknown Source)   
    6.      at chb.test.reflect.TestClass.testReflect(TestClass.java:20)<PRE></PRE>  

        解決方法:

    1. Class cls = Class.forName("chb.test.reflect.Student");   
    2. Object student = cls.newInstance();   
    3. Field field = cls.getDeclaredField("age");   
    4. field.setAccessible(true);//設置允許訪問   
    5. field.set(student, 10);   
    6. System.out.println(field.get(student));<PRE></PRE>  

        其實,在某些場合下(類中有get,set方法),可以先反射調用set方法,再反射調用get方法達到如上效果,代碼示例:

    1. Class cls = Class.forName("chb.test.reflect.Student");   
    2. Object student = cls.newInstance();   
    3.   
    4. Method setMethod = cls.getDeclaredMethod("setAge",Integer.class);   
    5. setMethod.invoke(student, 15);//調用set方法   
    6.                
    7. Method getMethod = cls.getDeclaredMethod("getAge");   
    8. System.out.println(getMethod.invoke(student));//再調用get方法<PRE></PRE>

    posted on 2009-01-11 17:28 forgood 閱讀(1207) 評論(1)  編輯  收藏 所屬分類: java

    評論

    # re: 反射機制補充 2009-01-11 17:52 forgood

    JAVA反射機制
    JAVA反射機制是在運行狀態中,對于任意一個類,都能夠知道這個類的所有屬性和方法;對于任意一個對象,都能夠調用它的任意一個方法;這種動態獲取的信息以及動態調用對象的方法的功能稱為java語言的反射機制。
    Java反射機制主要提供了以下功能: 在運行時判斷任意一個對象所屬的類;在運行時構造任意一個類的對象;在運行時判斷任意一個類所具有的成員變量和方法;在運行時調用任意一個對象的方法;生成動態代理。
    1. 得到某個對象的屬性

    1 public Object getProperty(Object owner, String fieldName) throws Exception {
    2 Class ownerClass = owner.getClass();
    3
    4 Field field = ownerClass.getField(fieldName);
    5
    6 Object property = field.get(owner);
    7
    8 return property;
    9 }
    Class ownerClass = owner.getClass():得到該對象的Class。

    Field field = ownerClass.getField(fieldName):通過Class得到類聲明的屬性。

    Object property = field.get(owner):通過對象得到該屬性的實例,如果這個屬性是非公有的,這里會報IllegalAccessException。

    2. 得到某個類的靜態屬性

    1 public Object getStaticProperty(String className, String fieldName)
    2 throws Exception {
    3 Class ownerClass = Class.forName(className);
    4
    5 Field field = ownerClass.getField(fieldName);
    6
    7 Object property = field.get(ownerClass);
    8
    9 return property;
    10 }

    Class ownerClass = Class.forName(className) :首先得到這個類的Class。

    Field field = ownerClass.getField(fieldName):和上面一樣,通過Class得到類聲明的屬性。

    Object property = field.get(ownerClass) :這里和上面有些不同,因為該屬性是靜態的,所以直接從類的Class里取。

    3. 執行某對象的方法

    1 public Object invokeMethod(Object owner, String methodName, Object[] args) throws Exception {
    2
    3 Class ownerClass = owner.getClass();
    4
    5 Class[] argsClass = new Class[args.length];
    6
    7 for (int i = 0, j = args.length; i < j; i++) {
    8 argsClass[i] = args[i].getClass();
    9 }
    10
    11 Method method = ownerClass.getMethod(methodName, argsClass);
    12
    13 return method.invoke(owner, args);
    14 }
    Class owner_class = owner.getClass() :首先還是必須得到這個對象的Class。

    5~9行:配置參數的Class數組,作為尋找Method的條件。

    Method method = ownerClass.getMethod(methodName, argsClass):通過Method名和參數的Class數組得到要執行的Method。

    method.invoke(owner, args):執行該Method,invoke方法的參數是執行這個方法的對象,和參數數組。返回值是Object,也既是該方法的返回值。

    4. 執行某個類的靜態方法

    1 public Object invokeStaticMethod(String className, String methodName,
    2 Object[] args) throws Exception {
    3 Class ownerClass = Class.forName(className);
    4
    5 Class[] argsClass = new Class[args.length];
    6
    7 for (int i = 0, j = args.length; i < j; i++) {
    8 argsClass[i] = args[i].getClass();
    9 }
    10
    11 Method method = ownerClass.getMethod(methodName, argsClass);
    12
    13 return method.invoke(null, args);
    14 }

    基本的原理和實例3相同,不同點是最后一行,invoke的一個參數是null,因為這是靜態方法,不需要借助實例運行。

    5. 新建實例
    1
    2 public Object newInstance(String className, Object[] args) throws Exception {
    3 Class newoneClass = Class.forName(className);
    4
    5 Class[] argsClass = new Class[args.length];
    6
    7 for (int i = 0, j = args.length; i < j; i++) {
    8 argsClass[i] = args[i].getClass();
    9 }
    10
    11 Constructor cons = newoneClass.getConstructor(argsClass);
    12
    13 return cons.newInstance(args);
    14
    15 }

    這里說的方法是執行帶參數的構造函數來新建實例的方法。如果不需要參數,可以直接使用newoneClass.newInstance()來實現。

    Class newoneClass = Class.forName(className):第一步,得到要構造的實例的Class。

    第5~第9行:得到參數的Class數組。

    Constructor cons = newoneClass.getConstructor(argsClass):得到構造子。

    cons.newInstance(args):新建實例。

    6. 判斷是否為某個類的實例

    1 public boolean isInstance(Object obj, Class cls) {
    2 return cls.isInstance(obj);
    3 }

    7. 得到數組中的某個元素
    1 public Object getByArray(Object array, int index) {
    2 return Array.get(array,index);
    3 }


      回復  更多評論   

    主站蜘蛛池模板: 美女被免费视频网站a| 亚洲国产精品VA在线观看麻豆| 性短视频在线观看免费不卡流畅| 中文字幕久精品免费视频| 亚洲免费在线观看| 在线视频网址免费播放| 一区二区三区在线观看免费| 免费一区二区三区在线视频| 一道本在线免费视频| 日本高清免费中文在线看| 美女扒开屁股让男人桶爽免费| 一级做a爰性色毛片免费| 一二三区免费视频| 最近国语视频在线观看免费播放 | 久久久久亚洲精品成人网小说| 亚洲精品成人网站在线观看| 亚洲国产一二三精品无码| 亚洲图片一区二区| 亚洲成人免费网址| 亚洲熟妇av午夜无码不卡| 337P日本欧洲亚洲大胆精品| 一级特黄a大片免费| 成人一区二区免费视频| 99免费在线观看视频| 99久久久精品免费观看国产| 毛片基地免费视频a| 国产免费卡一卡三卡乱码| 久久亚洲欧洲国产综合| 亚洲精品乱码久久久久久蜜桃不卡 | 国产成人3p视频免费观看| 亚洲а∨天堂久久精品| 中文字幕精品亚洲无线码二区 | 永久中文字幕免费视频网站| 国产成人精品日本亚洲专一区| 亚洲色大网站WWW永久网站| 色婷婷六月亚洲综合香蕉| a免费毛片在线播放| 91视频免费网址| 大陆一级毛片免费视频观看i| 亚洲色偷拍区另类无码专区| 亚洲AV永久无码精品成人|