首頁 >Java >java教程 >Java反射機制的原理和實作方法是什麼?

Java反射機制的原理和實作方法是什麼?

PHPz
PHPz轉載
2023-04-21 21:58:071273瀏覽

1.透過反射,我們可以建構實例,得到成員變數的值,得到方法並呼叫。

還可以得到定義在成員變數、方法、方法參數上的註解。

接下來看程式碼實現,然後講原理。

1)建構無參照實例:透過反射呼叫無參構造函數

        //1.通过全类名加载字节码对象
        Class clazz = Class.forName("com.example.lib.Person");
        //2.通过类的字节码拿到定义的构造函数
        Constructor constructor = clazz.getConstructor();
        //3.通过构造方法创建对象
        Object obj = constructor.newInstance();

2)建構有參數實例:

        //1.通过全类名加载字节码对象
        Class clazz = Class.forName("com.example.lib.Person");
        //2.通过类的字节码拿到定义的构造函数
        Constructor constructor = clazz.getConstructor(int.class,String.class);
        //3.通过构造方法创建对象
        Object obj = constructor.newInstance(20,"xiaohua");

3)透過反射來取得成員變數的值。

        //4.通过属性名获取属性
        Field field = clazz.getDeclaredField("age");
        field.setAccessible(true);
        //5.调用get方法拿到对象obj属性age的值
        Integer age = (Integer) field.get(obj);

4)透過反射呼叫物件的方法。

        //4.通过方法名和参数类型,拿到方法
        method = clazz.getMethod("setAge", int.class);
        //5.调用方法 obj是哪个对象身上的方法。
        method.invoke(obj, 21);
        method =  clazz.getMethod("getAge");
        method.invoke(obj);

5).透過反射取得靜態變數的值。

       //1.通过全类名加载字节码对象
        Class clazz = Class.forName("com.example.lib.Person");
        //2.获取静态属性ID
        Field  field = clazz.getField("ID");
        field.setAccessible(true);
        //3.拿到静态属性ID的值。
        // 因为静态变量存在方法区,在对象创建之前,就已经加装到了内存
        //所以,没有对象,也可以获取变量的值,这里传null也是可以的。
        int id = (int) field.get(null);

2.透過反射取得定義的註解的值

1)取得成員變數的註解以及值。

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface BindView {
    int value();
}
public class MainActivity {
    @BindView(10000)
    TextView textView;
}
        //10通过反射拿到定义在属性上的注解
        Class  clazz = MainActivity.class;
        Field textView = clazz.getDeclaredField("textView");
        BindView bindView = textView.getAnnotation(BindView.class);
        int txtId = bindView.value();

3)透過反射來取得定義在方法和方法參數上的註解以及值

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Post {
    String value() default "";
}
public interface NetWorkInterface {
    @Post("http://www.baidu.com")
    Call getPerson(@Queue("name") String name, @Queue("200") int price);
}
      //11通过反射拿到方法上定义的注解
        clazz = NetWorkInterface.class;
        Method method = clazz.getMethod("getPerson", String.class, int.class);
        //获取Post注解
        Post post = method.getAnnotation(Post.class);
        //获取值
        String url = post.value();
         //12通过反射拿到参数上的注解
        //为是个二维数组,因为方法参数会有多个,一个参数有可能定义多个注解
        Annotation[][] annotations = method.getParameterAnnotations();
        for (Annotation[] ans : annotations) {
            for (Annotation an : ans) {
                if (an instanceof Queue) {
                    Queue queue = (Queue) an;
                    String value = queue.value();
                }
            }
        }

4)來取得方法的參數和傳回值類型。

        //13.拿到方法参数的类型。
        Type[] types = method.getGenericParameterTypes();
        for (Type type : types) {
            System.out.println(type.toString());
        }
        //14.获取方法返回值类型
        Type type = method.getGenericReturnType();

3總結:透過反射,可以拿到物件身上的成員變數的值、呼叫方法,取得定義在成員變數、方法和 方法參數上的註解。 Retrofit 就用到了註解加反射技術,和動態代理(這個技術稍後分享)

4.透過反射,可以做到以上事情。反射的原理是啥?

1)我們寫的原始碼是.java文件,經過javac編譯後成為.class文件,即字節碼文件。

2)程式​​執行時,JVM會類別載入字節碼檔案到內存,嚴格意義上說是載入到方法區,並轉換成

java.lang.Class物件。萬事萬物皆對象,Class稱為類別對象,描述類別在元資料空間的資料結構,包含類別中定義的屬性、方法、建構方法、註解、介面等資訊。

所有反射的第一步是拿到類別物件Class物件。拿到了Class對象,也就拿到了類別中定義的一切。

Class clazz = Class.forName("com.example.lib.Person");

這行程式碼就是透過類別載入器把Person類別載入到內存,並得到對應的Class 物件。

以上是Java反射機制的原理和實作方法是什麼?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:yisu.com。如有侵權,請聯絡admin@php.cn刪除