Home > Java > javaTutorial > body text

What are the principles and implementation methods of Java reflection mechanism?

PHPz
Release: 2023-04-21 21:58:07
forward
1220 people have browsed it

1. Through reflection, we can build an instance, get the value of the member variable, get the method and call it.

You can also get annotations defined on member variables, methods, and method parameters.

Next, let’s look at the code implementation, and then talk about the principle.

1) Construct a parameterless instance: call the parameterless constructor through reflection

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

2) Construct a parameterized instance:

        //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");
Copy after login

3) Obtain the value of the member variable through reflection .

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

4) Call the object's method through reflection.

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

5). Obtain the value of static variables through reflection.

       //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);
Copy after login

2. Obtain the value of the defined annotation through reflection

1) Obtain the annotation and value of the member variable.

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

3) Obtain the annotations and values ​​defined on the method and method parameters through reflection

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Post {
    String value() default "";
}
Copy after login
public interface NetWorkInterface {
    @Post("http://www.baidu.com")
    Call getPerson(@Queue("name") String name, @Queue("200") int price);
}
Copy after login
      //11通过反射拿到方法上定义的注解
        clazz = NetWorkInterface.class;
        Method method = clazz.getMethod("getPerson", String.class, int.class);
        //获取Post注解
        Post post = method.getAnnotation(Post.class);
        //获取值
        String url = post.value();
Copy after login
         //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();
                }
            }
        }
Copy after login

4) Obtain the parameter and return value types of the method.

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

3 Summary: Through reflection, you can get the values ​​of member variables on the object, call methods, and obtain annotations defined on member variables, methods, and method parameters. Retrofit uses annotation plus reflection technology, and dynamic proxy (this technology will be shared later)

4. Through reflection, you can do the above things. What is the principle of reflection?

1) The source code we write is a .java file, which becomes a .class file after being compiled by javac, that is, a bytecode file.

2) When the program is executed, the JVM will class load the bytecode file into the memory. Strictly speaking, it is loaded into the method area and converted into a

java.lang.Class object. Everything is an object. Class is called a class object, which describes the data structure of a class in the metadata space, including the attributes, methods, constructors, annotations, interfaces and other information defined in the class.

The first step in all reflection is to get the class object Class object. Once you get the Class object, you also get everything defined in the class.

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

This line of code loads the Person class into memory through the class loader , and get the corresponding Class object.

The above is the detailed content of What are the principles and implementation methods of Java reflection mechanism?. For more information, please follow other related articles on the PHP Chinese website!

Related labels:
source:yisu.com
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template