Java语言高级特性——注解与反射(二)

一 、反射简介

一般情况下,我们使用某个类时必定知道它是什么类,是用来做什么的,并且能够获得此类的引用。于是我们直接对这个类进行实例化,之后使用这个类对象进行操作。
反射则是一开始并不知道我要初始化的类对象是什么,自然也无法使用 new 关键字来创建对象了。
反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和
方法;对于任意一个对象,都能够调用它的任意方法和属性;并且能改变它的属性。

二、Java反射机制提供的功能

  1. 在运行时构造任意一个类的对象
  2. 在运行时获取或者修改任意一个类所具有的成员变量和方法
  3. 在运行时调用任意一个对象的方法(属性)

三、Class 相关属性和方法

反射始于Class,
Class是一个类,封装了当前对象所对应的类的信息
Class类是一个对象照镜子的结果,对象可以看到自己有哪些属性,方法,构造器,实现了哪些接口等等。对于每个类而言,JRE 都为其保留一个不变的 Class 类型的对象。一个 Class 对象包含了特定某个类的有关信息。 对象只
能由系统建立对象,一个类(而不是一个对象)在 JVM 中只会有一个Class实例。

1. 获得 Class 对象

获取Class对象的三种方式

  1. 通过类名获取 类名.class
  2. 通过对象获取 对象名.getClass()
  3. 通过全类名获取 Class.forName(全类名) classLoader.loadClass(全类名)

2. 判断是否为某个类的实例

一般地,我们用 instanceof 关键字来判断是否为某个类的实例。同时我们也可以借助反射中 Class 对象的isInstance() 方法来判断是否为某个类的实例,它是一个 native 方法:

3. 创建实例

通过反射来生成对象主要有两种方式。

  • 使用Class对象的newInstance()方法来创建Class对象对应类的实例。
Class c = String.class;
Object str = c.newInstance();
  • 先通过Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建实例。这
    种方法可以用指定的构造器构造类的实例。
//获取String所对应的Class对象
Class c = String.class;
//获取String类带一个String参数的构造器
Constructor constructor = c.getConstructor(String.class);
//根据构造器创建实例
Object obj = constructor.newInstance("23333");
System.out.println(obj);

4. 获取构造器信息

得到构造器的方法

Constructor getConstructor(Class[] params) -- 获得使用特殊的参数类型的public构造函数(包括父类)
Constructor[] getConstructors() -- 获得类的所有公共构造函数
Constructor getDeclaredConstructor(Class[] params) -- 获得使用特定参数类型的构造函数(包括私有)
Constructor[] getDeclaredConstructors() -- 获得类的所有构造函数(与接入级别无关)

5. 获取类的成员变量(字段)信息

获得字段信息的方法

Field getField(String name) -- 获得命名的公共字段
Field[] getFields() -- 获得类的所有公共字段
Field getDeclaredField(String name) -- 获得类声明的命名的字段
Field[] getDeclaredFields() -- 获得类声明的所有字段

6.调用方法

Method getMethod(String name, Class[] params) -- 使用特定的参数类型,获得命名的公共方法
Method[] getMethods() -- 获得类的所有公共方法
Method getDeclaredMethod(String name, Class[] params) -- 使用特写的参数类型,获得类声明的命名的方法
Method[] getDeclaredMethods() -- 获得类声明的所有方法

当我们从类中获取了一个方法后,我们就可以用 invoke() 方法来调用这个方法。invoke 方法的原型为:

public Object invoke(Object obj, Object... args)

7.利用反射创建数组

数组在Java里是比较特殊的一种类型,它可以赋值给一个Object Reference 其中的Array类为
java.lang.reflect.Array类。我们通过Array.newInstance()创建数组对象,它的原型是:

public static Object newInstance(Class componentType, int length);

8.反射获取泛型真实类型

当我们对一个泛型类进行反射时,需要的到泛型中的真实数据类型,来完成如json反序列化的操作。此时需要通
过Type 体系来完成。Type 接口包含了一个实现类(Class)和四个实现接口,他们分别是:

  • TypeVariable
    泛型类型变量。可以泛型上下限等信息;
  • ParameterizedType
    具体的泛型类型,可以获得元数据中泛型签名类型(泛型真实类型)
  • GenericArrayType
    当需要描述的类型是泛型类的数组时,比如List[],Map[],此接口会作为Type的实现。
  • WildcardType
    通配符泛型,获得上下限信息;

两个小栗子

1. 反序列化

public class MyClass {
    static class Response {
        T data;
        int code;
        String message;

        @Override
        public String toString() {
            return "Response{" +
                    "data=" + data +
                    ", code=" + code +
                    ", message='" + message + '\'' +
                    '}';
        }

        public Response(T data, int code, String message) {

            this.data = data;
            this.code = code;
            this.message = message;
        }
    }

    static class Data {
        String result;

        public Data(String result) {
            this.result = result;
        }

        @Override
        public String toString() {
            return "Data{" +
                    "result=" + result +
                    '}';
        }
    }




    public static void main(String[] args) {
        //序列化
        Response dataResponse = new Response(new Data("数据"), 1, "成功");
        Gson gson = new Gson();
        String json = gson.toJson(dataResponse);
        System.out.println(json);
        //反序列化
        //1.使用Gosn提供的TypeToken 获取泛型的真实类型
        Type type1 = new TypeToken>(){}.getType();

        Type type2 = new TypeRefrence>(){}.getType();


        Response response = gson.fromJson(json,type2);
        System.out.println(response.data.getClass());
    }


}

public class TypeRefrence {
    Type type;
    T t;
    protected TypeRefrence() {
        //获得泛型类型
        Type genericSuperclass = getClass().getGenericSuperclass();
        ParameterizedType parameterizedType = (ParameterizedType) genericSuperclass;
        //因为类泛型可以定义多个  A 所以是个数组
        Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
        type = actualTypeArguments[0];
    }

    public Type getType() {
        return type;
    }
}

2. 仿Butterknife功能

  • 首先定义一个注解 来标记View
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectView {
    @IdRes int value();
}
  • 其次是通过反射来处理运行时的操作 findViewById()
package fastec.com.rrcc.projectx1.inject;

import android.app.Activity;
import android.view.View;

import java.lang.reflect.Field;

import fastec.com.rrcc.projectx1.annotation.InjectView;

/**
 * @ProjectName: project
 * @Package: fastec.com.rrcc.projectx1.inject
 * @ClassName: InjectUtil
 * @Description: java类作用描述
 * @Author: haoran.li
 * @CreateDate: 2020/5/6 上午9:22
 * @UpdateUser: haoran.li
 * @UpdateDate: 2020/5/6 上午9:22
 * @UpdateRemark: 更新说明
 * @Version: 1.0
 */
public class InjectUtil {

    public static void inject(Activity activity) {
        Class cls = activity.getClass();
        //获得此类所有的成员
        Field[] fields = cls.getDeclaredFields();

        //遍历带注解的所有字段对象 并处理
        for (Field field : fields) {
            if (field.isAnnotationPresent(InjectView.class) == true) {
                InjectView injectView = field.getAnnotation(InjectView.class);
                //通过反射来获取值id
                int id = injectView.value();
                View view = activity.findViewById(id);
                //反射设置 属性的值
                field.setAccessible(true);
                try {
                    //反射赋值
                    field.set(activity, view);
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }

            }
        }

    }
}

  • 最后就是在Activity中使用验证
package fastec.com.rrcc.projectx1;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;

import fastec.com.rrcc.projectx1.annotation.InjectView;
import fastec.com.rrcc.projectx1.inject.InjectUtil;


public class MainActivity extends AppCompatActivity {

    @InjectView(R.id.test)
    TextView text;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        InjectUtil.inject(MainActivity.this);

        text.setText("测试注解反射");

    }
}

完整代码

你可能感兴趣的:(Java语言高级特性——注解与反射(二))