java的反射技術

  java的反射機制:

          我們都知道,一個java類中,有成員變量、成員方法、構造方法等等,反射機制,就是動態的獲取這些類中的信息,並動態的調用對象的方法。通常只有要設計框架的過程中,纔會使用java反射相關的api,但是瞭解java反射機制的基本原理與基本方法,可以幫忙我們在日後對框架的學習與應用提供很大的幫忙。

 

     瞭解Class類:

          Class類代表着某個類的字節碼,要使用反射,就需要取得對應的Class對象,然後就通過這個對象,就可解剖出類的成員變量,成員方法等等。

  Class類的常用方法:

  getConstructor() 獲取構造函數

  getMethod()  獲取成員方法

  getField() 獲取成員變量

   

  getDeclaredConstructor() 獲取私有的構造函數

  getDeclaredMethod()  獲取私有的成員方法

  getDeclaredField() 獲取私有的成員變量

 

     如何獲取Class類對象:

[java]  

//通過Class的forName()方法,此方法最爲常用  

Class class1 = Class.forName("com.java4fun.reflect.Person");  

//通過 Person.class  

Class class2 = Person.class;  

//通過對象獲得  

Class class3 = new Person().getClass();  

 

反射的使用,簡單的講,就是通過類的Class對象,獲取對應的Field、Method 和 Constructor 對象,並進行相關操作。

下面就用反射來解剖Person類,給出Person類:

[java]  

public class Person {  

      

    public String name = "java";  

    private int age = 2013;  

      

    //無參構造函數  

    public Person() {  

    }  

      

    public Person(int age) {  

        super();  

        this.age = age;  

    }  

    //私有構造函數  

    private Person(String name) {  

        super();  

        this.name = name;  

    }  

  

    public Person(String name, int age) {  

        super();  

        this.name = name;  

        this.age = age;  

    }  

  

    public int getAge() {  

        return age;  

    }  

  

    public String getName() {  

        return name;  

    }  

  

    public void setAge(int age) {  

        this.age = age;  

    }  

  

    public void setName(String name) {  

        this.name = name;  

    }  

  

    public static void getMessage(String message){  

        System.out.println(message);  

    }  

      

    public static void getFriends(String friends[]){  

        for(String s:friends){  

            System.out.print(s+"\t");  

        }  

    }  

      

    @Override  

    public String toString() {  

        return "Person [name=" + name + ", age=" + age + "]";  

    }  

      

    private void secret(String secret){  

        System.out.println(secret);  

    }  

      

    public void run(int num){  

        System.out.println(num);  

    }  

}  

 

 

 

構造函數的獲取:

[java]  

import java.lang.reflect.Constructor;  

  

public class ConstructorTest {  

    // 利用反射技術,完成對Person.類的相關操作  

    public static void main(String[] args) throws Exception {  

         constructor1(); //無參構造函數的獲取與使用  

         constructor2(); // 帶參數的構造函數的獲取使用  

         constructor3(); // 私有的構造函數的獲取使用  

    }  

  

    // 獲取私有構造函數  

    private static void constructor3() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲取私有構造函數對象。  

        Constructor c = clazz.getDeclaredConstructor(String.class);  

        // 由於私有的構造函數無數直接構造對象,需要調用setAccessible(),  

        //此方法來自於AccessibleObject 類,它是 Field、Method 和 Constructor 對象的基類  

        c.setAccessible(true);  

        // 利用這個構造函數,構造一個Person對象  

        Person p = (Person) c.newInstance("private constructor");  

        System.out.println(p);  

    }  

  

    // 訪問帶參數的構造函數  

    private static void constructor2() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲取帶參數的構造函數對象。  

        Constructor c = clazz.getConstructor(String.class, int.class);  

        // 利用這個構造函數,構造一個Person對象  

        Person p = (Person) c.newInstance("hello", 110);  

        System.out.println(p);  

    }  

  

    // 訪問無參構造函數  

    private static void constructor1() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲取無參的構造函數對象。  

        Constructor c = clazz.getConstructor(null);  

        // 利用這個構造函數,構造一個Person對象  

        Person p = (Person) c.newInstance(null);  

        System.out.println(p);  

  

        // Class類也提供了一個可直接用無參構造函數構造對象的方法  

        // Person person = (Person) clazz.newInstance();  

        // System.out.println(person);  

    }  

}  

 

成員方法的獲取:

[java] 

import java.lang.reflect.Method;  

  

public class MethodTest {  

    // 利用反射技術,完成對Person.類的相關操作  

    public static void main(String[] args) throws Exception {  

         //method1(); //無參成員方法的獲取與使用  

        // method2(); // 帶參數成員方法的獲取使用  

         //method3(); // 私有成員方法的獲取使用  

        method4(); // 參數是數組的方法解決方法  

    }  

  

    // 獲取無參成員方法  

    private static void method1() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲取無參的方法, 反射Person類的中 public String getName(){}方法  

        Method m = clazz.getMethod("getName", null);  

        //運行這個方法,由於方法的運行需要對象,爲了方便,在這裏直接通過傳統的方法創建一個對象。  

        Person p = new Person("HelloWorld",23);  

        String name = (String) m.invoke(p, null);  

        System.out.println(name);  

    }  

      

    // 獲取帶參數的成員方法  

    private static void method2() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲取帶參數的方法, 反射Person類的中 public void run(int num){}方法  

        Method m = clazz.getMethod("run", int.class);  

        //運行這個方法,由於方法的運行需要對象,爲了方便,在這裏直接通過傳統的方法創建一個對象。  

        Person p = new Person("HelloWorld",23);  

        m.invoke(p, 11111);  

    }  

      

    // 獲取私有的成員方法  

    private static void method3() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲私有的方法, 反射Person類的中 private void secret(String secret){}方法  

        Method m = clazz.getDeclaredMethod("secret", String.class);  

        //讓私有方法可以訪問  

        m.setAccessible(true);  

        //運行這個方法,由於方法的運行需要對象,爲了方便,在這裏直接通過傳統的方法創建一個對象。  

        Person p = new Person("HelloWorld",23);  

        m.invoke(p, "最近變胖了");  

    }  

  

    // 參數是數組的方法解決方法  

    private static void method4() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲靜態方法, 反射Person類的中 public static void getFriends(String friends[]){}方法  

        Method m = clazz.getMethod("getFriends", String[].class);  

          

        //這樣調用方法會出錯,由於版本的遺留問題,new String[]{"java","c","c++"}會被認爲是"java" "c"  "c++"  

        //m.invoke(null, new String[]{"java","c","c++"});  

        //相當於getFriends("java","c","c++");  

          

        //正確調用方法,相當於把數組當成一個對象傳入  

        m.invoke(null, (Object)new String[]{"java","c","c++"});  

    }  

}  

 

成員屬性的獲取:

[java]  

import java.lang.reflect.Field;  

  

public class FieldTest {  

    // 利用反射技術,完成對Person.類的相關操作  

    public static void main(String[] args) throws Exception {  

         method1(); // 獲取屬性  

         method2(); // 獲取私有屬性  

    }  

  

    // 獲取屬性  

    private static void method1() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲取屬性, 反射Person類的中 public String name = "java";屬性  

        Field f = clazz.getField("name");  

        // 使用屬性,需指定對象,爲了方便,在這裏直接通過傳統的方法創建一個對象。  

        Person p = new Person();  

        Class type = f.getType();  

        if (type.equals(String.class)) {  

            String name = (String) f.get(p);  

            System.out.println(name);  

        }  

    }  

  

    // 獲取私有屬性    

    private static void method2() throws Exception {  

        // 獲取Person的字節碼對象。  

        Class clazz = Class.forName("com.java4fun.reflect.Person");  

        // 獲取私有屬性, 反射Person類的中 private int age = 2013;屬性  

        Field f = clazz.getDeclaredField("age");  

        f.setAccessible(true);  

        // 使用屬性,需指定對象,爲了方便,在這裏直接通過傳統的方法創建一個對象。  

        Person p = new Person();  

        Class type = f.getType();  

        if (type.equals(int.class)) {  

            int age = f.getInt(p);  

            System.out.println(age);  

        }  

    }  

}  

 


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章