第十章 反射机制

反射机制概述

Java反射机制是指在运行时动态获取类的信息或动态调用对象的方法、修改属性等操作。
主要核心就是Class类、Constructor类、Field类、Method类等API。 
反射机制主要应用于框架开发、动态代理、ORM框架、JDBC驱动等方面。
通过反射机制,程序员能够获得在编译期间不被知晓的类、属性、方法等信息。
但是反射机制的性能较低,常常被认为是一种牺牲性能换取灵活性的实现方式。

Java反射机制核心包:java.lang.reflect.*

Java反射机制核心类:
java.lang.Class
java.lang.reflect.Field
java.lang.reflect.Method
java.lang.reflect.Constructor
java.lang.reflect.Modifier

获取Class

该过程发生了类加载

Java中获取Class对象有以下三种方式:

1. 调用Object类的getClass()方法
可以通过对象的getClass()方法来获取Class对象,例如:
Object obj = new Object();
Class clazz = obj.getClass();

2. 使用“类.class”语法
可以使用“类.class”语法来获取Class对象,例如:
Class clazz = Object.class;

3. 使用Class类的forName()方法
可以使用Class类的forName()方法来获取Class对象,例如:
Class clazz = Class.forName("java.lang.Object");
/**
 * 1. 反射机制是JDK中的一套类库,这套类库可以帮助我们操作/读取 class 字节码文件。
 * 2. 后期学习的大量的java框架,底层都是基于反射机制实现的,所以必须掌握(要能够数量的使用反射机制中的方法)。
 * 3. 反射机制可以让程序更加灵活。怎么灵活????
 * 4. 反射机制最核心的几个类:
 *      java.lang.Class:Class类型的实例代表硬盘上的某个class文件。或者说代表某一种类型。
 *      java.lang.reflect.Filed:Filed类型的实例代表类中的属性/字段
 *      java.lang.reflect.Constructor: Constructor类型的实例代表类中的构造方法
 *      java.lang.reflect.Method: Method类型的实例代表类中的方法
 * 5. 在java语言中获取Class的三种方式:
 *      第一种方式:Class c = Class.forName("完整的全限定类名");
 *          注意:
 *              1.全限定类名是带有包名的。
 *              2.是lang包下的,java.lang也不能省略。
 *              3.这是个字符串参数。
 *              4.如果这个类根本不存在,运行时会报异常:java.lang.ClassNotFoundException
 *              5.这个方法的执行会导致类的加载动作的发生。
 *      第二种方式:Class c = obj.getClass();
 *          注意:这个方法是通过引用去调用的。
 *      第三种方式:在java语言中,任何一种类型,包括基本数据类型,都有 .class 属性。用这个属性可以获取Class实例。
 */

public class ReflectTest01 {
    public static void main(String[] args) throws ClassNotFoundException {

        // stringClass 就代表 String类型。
        // stringClass 就代表硬盘上的 String.class文件。
        Class stringClass = Class.forName("java.lang.String");

        // 获取 User 类型
        Class userClass = Class.forName("com.powernode.javase.reflect.User");

        String s1 = "动力节点";
        Class stringClass2 = s1.getClass();

        // 某种类型的字节码文件在内存当中只有一份。
        // stringClass 和 stringClass2 都代表了同一种类型:String类型
        System.out.println(stringClass == stringClass2); // true

        User user = new User("zhangsan", 20);
        Class userClass2 = user.getClass();
        System.out.println(userClass2 == userClass); // true

        // intClass 代表的就是基本数据类型 int类型
        Class intClass = int.class;
        Class doubleClass = double.class;
        Class stringClass3 = String.class;
        Class userClass3 = User.class;

        System.out.println(stringClass3 == stringClass); // true
    }
}
/**
 * 获取到Class之后能干啥?
 *      1. 至少可以实例化对象。
 */
public class ReflectTest02 {
    public static void main(String[] args) throws Exception{

        // 获取到Class类型的实例之后,可以实例化对象
        // 通过反射机制实例化对象
        Class userClass = Class.forName("com.powernode.javase.reflect.User"); // userClass 代表的就是 User类型。

        // 通过userClass来实例化User类型的对象
        // 底层实现原理是:调用了User类的无参数构造方法完成了对象的实例化。
        // 要使用这个方法实例化对象的话,必须保证这个类中是存在无参数构造方法的。如果没有无参数构造方法,则出现异常:java.lang.InstantiationException
        User user = (User)userClass.newInstance();

        System.out.println(user);

        User user2 = (User)userClass.newInstance();
        System.out.println(user == user2); // false
    }
}

反射Field

反射Field包括两方面:

一方面:通过反射机制获取Field

    Field field = clazz.getDeclaredField("fieldName"); // 通过属性名获取 

    Field[] fields = clazz.getDeclaredFields(); // 获取所有的属性,包括私有的

另一方面:通过Filed访问对象的属性

    Object fieldValue = field.get(myObject); // 读取某个对象的属性值

    field.set(myObject, newValue); // 修改某个对象的属性值

/**
 * 关于反射机制中的 java.lang.reflect.Field(代表的是一个类中的字段/属性)
 */
public class ReflectTest04 {
    public static void main(String[] args) throws Exception{
        // 获取Vip类
        Class vipClass = Class.forName("com.powernode.javase.reflect.Vip");

        /*// 获取Vip类中所有 public 修饰的属性/字段
        Field[] fields = vipClass.getFields();
        System.out.println(fields.length);

        // 遍历数组
        for(Field field : fields){
            System.out.println(field.getName());
        }*/

        // 获取Vip类中所有的属性/字段,包括私有的
        Field[] fields = vipClass.getDeclaredFields();

        for(Field field : fields){
            // 获取属性名
            System.out.println(field.getName());
            // 获取属性类型
            Class fieldType = field.getType();
            // 获取属性类型的简单名称
            System.out.println(fieldType.getSimpleName());
            // 获取属性的修饰符
            //System.out.println(field.getModifiers());
            System.out.println(Modifier.toString(field.getModifiers()));
        }
    }
}

反编译类中所有的属性

// 获取当前类的父类
sb.append(stringClass.getSuperclass().getName());
// 获取当前类的实现的所有接口
Class[] interfaces = stringClass.getInterfaces();
public static void main(String[] args) throws Exception{
    // 获取String类
    Class stringClass = Class.forName("com.powernode.javase.reflect.Vip");

    // 字符串拼接
    StringBuilder sb = new StringBuilder();

    // 获取类的修饰符
    sb.append(Modifier.toString(stringClass.getModifiers()));

    sb.append(" class ");

    //sb.append(stringClass.getSimpleName());
    sb.append(stringClass.getName());

    sb.append(" extends ");

    // 获取当前类的父类
    sb.append(stringClass.getSuperclass().getName());

    // 获取当前类的实现的所有接口
    Class[] interfaces = stringClass.getInterfaces();
    if(interfaces.length > 0){
        sb.append(" implements ");
        for (int i = 0; i < interfaces.length; i++) {
            Class interfaceClass = interfaces[i];
            sb.append(interfaceClass.getName());
            if(i != interfaces.length - 1){
                sb.append(",");
            }
        }
    }

    sb.append("{\n");

    // 获取所有属性
    Field[] fields = stringClass.getDeclaredFields();
    for (Field field : fields){
        sb.append("\t");
        sb.append(Modifier.toString(field.getModifiers()));
        sb.append(" ");
        sb.append(field.getType().getName());
        sb.append(" ");
        sb.append(field.getName());
        sb.append(";\n");
    }

    sb.append("}");

    // 输出
    System.out.println(sb);
}

通过反射为对象属性赋值

// 调用方法打破封装

ageField.setAccessible(true);

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

    // 如果不使用反射机制,怎么访问对象的属性?
    Customer customer = new Customer();

    // 修改属性的值(set动作)
    // customer要素一
    // name要素二
    // "张三"要素三
    customer.name = "张三";

    // 读取属性的值(get动作)
    // 读取哪个对象的哪个属性
    System.out.println(customer.name);

    // 如果使用反射机制,怎么访问对象的属性?
    // 获取类
    Class clazz = Class.forName("com.powernode.javase.reflect.Customer");

    // 获取对应的Field
    Field ageField = clazz.getDeclaredField("age");

    // 调用方法打破封装
    ageField.setAccessible(true);

    // 修改属性的值
    // 给对象属性赋值三要素:给哪个对象 的 哪个属性 赋什么值
    ageField.set(customer, 30);

    // 读取属性的值
    System.out.println("年龄:" + ageField.get(customer));

    // 通过反射机制给name属性赋值,和读取name属性的值
    Field nameField = clazz.getDeclaredField("name");
    // 修改属性name的值
    nameField.set(customer, "李四");
    // 读取属性name的值
    System.out.println(nameField.get(customer));


}

反射Method

反射Method包括两方面:

一方面:通过反射机制获取Method

    Method method = clazz.getDeclaredMethod("methodName", paramTypes);

另一方面:通过Method调用方法

    Class clazz = MyClass.class;
    Method method = clazz.getDeclaredMethod("methodName", paramTypes);
    Object[] args = {arg1, arg2, arg3};
    Object result = method.invoke(myObject, args);

/**
 * 通过反射机制获取Method
 */
public class ReflectTest08 {
    public static void main(String[] args) throws Exception{
        // 获取类
        Class clazz = Class.forName("com.powernode.javase.reflect.UserService");

        // 获取所有的方法,包含私有的方法
        Method[] methods = clazz.getDeclaredMethods();
        //System.out.println(methods.length);

        // 遍历数组
        for(Method method : methods){
            // 方法修饰符
            System.out.println(Modifier.toString(method.getModifiers()));
            // 方法返回值类型
            System.out.println(method.getReturnType().getName());
            // 方法名
            System.out.println(method.getName());
            // 方法的参数列表
            /*Class<?>[] parameterTypes = method.getParameterTypes();
            for (Class parameterType : parameterTypes){
                System.out.println(parameterType.getName());
            }*/

            Parameter[] parameters = method.getParameters();
            for (Parameter parameter : parameters){
                System.out.println(parameter.getType().getName());
                System.out.println(parameter.getName());// arg0, arg1, arg2......
            }
        }
    }
}

反编译类的方法

// 获取所有的方法
Method[] methods = stringClass.getDeclaredMethods();

/**
 * 反射一个类中所有的方法,然后进行拼接字符串。
 */
public class ReflectTest09 {
    public static void main(String[] args) throws Exception{
        StringBuilder sb = new StringBuilder();
        Class stringClass = Class.forName("com.powernode.javase.reflect.UserService");
        // 获取类的修饰符
        sb.append(Modifier.toString(stringClass.getModifiers()));
        sb.append(" class ");
        // 获取类名
        sb.append(stringClass.getName());
        // 获取父类名
        sb.append(" extends ");
        sb.append(stringClass.getSuperclass().getName());
        // 获取父接口名
        Class[] interfaces = stringClass.getInterfaces();
        if(interfaces.length > 0){
            sb.append(" implements ");
            for (int i = 0; i < interfaces.length; i++) {
                sb.append(interfaces[i].getName());
                if(i != interfaces.length - 1){
                    sb.append(",");
                }
            }
        }
        sb.append("{\n");

        // 类体
        // 获取所有的方法
        Method[] methods = stringClass.getDeclaredMethods();
        for(Method method : methods){
            sb.append("\t");
            // 追加修饰符
            sb.append(Modifier.toString(method.getModifiers()));
            // 追加返回值类型
            sb.append(" ");
            sb.append(method.getReturnType().getName());
            // 追加方法名
            sb.append(" ");
            sb.append(method.getName());
            sb.append("(");
            // 追加参数列表
            Parameter[] parameters = method.getParameters();
            for (int i = 0; i < parameters.length; i++) {
                Parameter parameter = parameters[i];
                sb.append(parameter.getType().getName());
                sb.append(" ");
                sb.append(parameter.getName());
                if(i != parameters.length - 1){
                    sb.append(",");
                }
            }
            sb.append("){}\n");
        }

        sb.append("}");

        // 输出
        System.out.println(sb);

    }
}

通过反射机制调用方法

/**
 * 通过反射机制调用Method
 */
public class ReflectTest10 {
    public static void main(String[] args) throws Exception {
        // 不使用反射机制怎么调用方法?
        // 创建对象
        UserService userService = new UserService();

        // 调用方法
        // 分析:调用一个方法需要几个要素?四要素
        // 调用哪个对象的哪个方法,传什么参数,返回什么值
        boolean isSuccess = userService.login("admin", "123456");
        System.out.println(isSuccess ? "登录成功" : "登录失败");

        // 调用方法
        userService.logout();

        // 通过反射机制调用login方法
        // 获取Class
        Class clazz = Class.forName("com.powernode.javase.reflect.UserService");

        // 获取login方法
        Method loginMethod = clazz.getDeclaredMethod("login", String.class, String.class);

        // 调用login方法
        Object retValue = loginMethod.invoke(userService, "admin", "123456");
        System.out.println(retValue);

        // 调用logout方法
        Method logoutMethod = clazz.getDeclaredMethod("logout");
        logoutMethod.invoke(userService);
    }
}

反射Constructor

反射Constructor包括两方面:

一方面:通过反射机制获取Constructor

Constructor constructor2 = clazz.getDeclaredConstructor(paramTypes); 

另一方面:通过Constructor创建对象

Class clazz = MyClass.class;
Constructor constructor = clazz.getDeclaredConstructor(paramTypes);
Object[] args = {arg1, arg2, arg3};
Object myObject = constructor.newInstance(args);

// 获取所有的构造方法
Constructor[] cons = clazz.getDeclaredConstructors();

/**
 * 通过反射机制获取一个类中所有的构造方法
 */
public class ReflectTest11 {
    public static void main(String[] args) throws Exception{
        StringBuilder sb = new StringBuilder();
        // 获取类
        Class clazz = Class.forName("java.lang.String");
        // 类的修饰符
        sb.append(Modifier.toString(clazz.getModifiers()));
        sb.append(" class ");
        // 类名
        sb.append(clazz.getName());
        sb.append(" extends ");
        // 父类名
        sb.append(clazz.getSuperclass().getName());
        // 实现的接口
        Class[] interfaces = clazz.getInterfaces();
        if(interfaces.length > 0) {
            sb.append(" implements ");
            for (int i = 0; i < interfaces.length; i++) {
                sb.append(interfaces[i].getName());
                if(i != interfaces.length - 1){
                    sb.append(",");
                }
            }
        }
        sb.append("{\n");

        //类体
        // 获取所有的构造方法
        Constructor[] cons = clazz.getDeclaredConstructors();
        // 遍历所有的构造方法
        for(Constructor con : cons){
            sb.append("\t");
            // 构造方法修饰符
            sb.append(Modifier.toString(con.getModifiers()));
            sb.append(" ");
            // 构造方法名
            sb.append(con.getName());
            sb.append("(");

            // 构造方法参数列表
            Parameter[] parameters = con.getParameters();
            for (int i = 0; i < parameters.length; i++) {
                Parameter parameter = parameters[i];
                sb.append(parameter.getType().getName());
                sb.append(" ");
                sb.append(parameter.getName());
                if(i != parameters.length - 1){
                    sb.append(",");
                }
            }

            sb.append("){}\n");
        }

        sb.append("}");

        System.out.println(sb);
    }
}

通过反射机制调用构造方法来创建对象

/**
 * 通过反射机制调用构造方法来创建对象。
 */
public class ReflectTest12 {
    public static void main(String[] args) throws Exception{

        // 不使用反射机制的时候,怎么创建的对象?
        Order order1 = new Order();
        System.out.println(order1);

        Order order2 = new Order("1111122222", 3650.5, "已完成");
        System.out.println(order2);

        // 通过反射机制来实例化对象?
        Class clazz = Class.forName("com.powernode.javase.reflect.Order");
        // 这种方式依赖的是必须有一个无参数构造方法。如果没有会出现异常!
        // 在Java9的时候,这个方法被标注了已过时。不建议使用了。
        /*Object obj = clazz.newInstance();
        System.out.println(obj);*/

        // 获取Order的无参数构造方法
        Constructor defaultCon = clazz.getDeclaredConstructor();
        // 调用无参数构造方法实例化对象
        Object obj = defaultCon.newInstance();
        System.out.println(obj);

        // 获取三个参数的构造方法
        Constructor threeArgsCon = clazz.getDeclaredConstructor(String.class, double.class, String.class);
        // 调用三个参数的构造方法
        Object obj1 = threeArgsCon.newInstance("5552454222", 6985.0, "未完成");
        System.out.println(obj1);
    }
}

模拟框架

/**
 * 模拟框架的部分代码。通过读取属性配置文件,获取类信息,方法信息,然后通过反射机制调用方法。
 */
public class ReflectTest13 {
    public static void main(String[] args) throws Exception {
        // 读取属性配置文件
        ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.javase.reflect.config");
        String className = bundle.getString("className");
        String methodName = bundle.getString("methodName");
        String parameterTypes = bundle.getString("parameterTypes");
        String parameterValues = bundle.getString("parameterValues");

        // 通过反射机制调用方法
        // 创建对象(依赖无参数构造方法)
        Class<?> clazz = Class.forName(className);
        Constructor<?> defaultCon = clazz.getDeclaredConstructor();
        Object obj = defaultCon.newInstance();

        // 获取方法
        // java.lang.String,java.lang.String
        String[] strParameterTypes = parameterTypes.split(",");
        Class[] classParameterTypes = new Class[strParameterTypes.length];
        for (int i = 0; i < strParameterTypes.length; i++) {
            classParameterTypes[i] = Class.forName(strParameterTypes[i]);
        }
        Method method = clazz.getDeclaredMethod(methodName, classParameterTypes);

        // 调用方法
        // parameterValues=admin,123456
        Object retValue = method.invoke(obj, parameterValues.split(","));

        System.out.println(retValue);
    }
}

类加载及双亲委派机制

类加载的过程

  1. 装载(loading)
    1. 类加载器负责将类的class文件读入内存,并创建一个java.lang.Class对象
  1. 连接(linking)
    1. 验证(Verify)
      1. 确保加载类的信息符合JVM规范。
    1. 准备(Prepare)
      1. 正式为静态变量在方法区中开辟存储空间并设置默认值
      2. public static int k = 10; 此时:k会赋值0
      3. public static final int f = 10; 此时: f会赋值10
    1. 解析(Resolve)
      1. 将虚拟机常量池内的符号引用替换为直接引用(地址)的过程。
  1. 初始化(initialization)
    1. 静态变量赋值,静态代码块执行

低版本的JDK中类加载器的名字:

启动类加载器:负责加载rt.jar

扩展类加载器:ext/*.jar

系统类加载器:classpath

获取Class的四种方式

  1. 静态方法
    1. Class clazz = Class.forName(“全限定类名”)
  1. 实例方法
    1. Class clazz = 引用.getClass();
  1. class属性
    1. Class clazz = 类型名.class;
  1. 通过类加载器获取
    1. ClassLoader classLoader = ClassLoader.getSystemClassLoader();
    2. Class clazz = classLoader.loadClass(“全限定类名”);

Class.forName和classLoader.loadClass()的区别?

  • Class.forName():类加载时会进行初始化。
  • classLoader.loadClass():类加载时不会进行初始化,直到第一次使用该类。
/**
 * 获取Class的第四种方式
 */
public class ReflectTest14 {
    public static void main(String[] args) throws Exception {
        // 获取类加载器对象(获取的是 系统类加载器/应用类加载器 )
        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();

        // jdk.internal.loader.ClassLoaders$AppClassLoader@36baf30c
        // 这个类加载器是负责加载 classpath 中的字节码文件的。
        //System.out.println(systemClassLoader);

        // 加载类:但是这个加载过程只是将类加载过程中的前两步完成了,第三步的初始化没做。
        // 什么时候做初始化?在这个类真正的被第一次使用的时候。
        Class<?> aClass = systemClassLoader.loadClass("com.powernode.javase.reflect.User");

        System.out.println(aClass.newInstance());

        // 这种方式会走完类加载的全部过程,三步齐全
        //Class clazz = Class.forName("com.powernode.javase.reflect.User");

    }
}

类加载器

  1. 虚拟机内部提供了三种类加载器(Java9+):
    1. 启动类加载器(BootstrapClassLoader):加载Java最核心的类,例如String
    2. 平台类加载器(PlatformClassLoader):加载Java平台扩展的类库,例如解析XML的
    3. 应用类加载器(AppClassLoader):加载classpath中的
    4. 同时我们还可以自定义一个类加载器(UserClassLoader)
  1. 获取类加载器可以通过 getParent()方法一级一级获取
/**
 * 虚拟机内部有三个不同的类加载器:
 *      1. 启动类加载器:BootstrapClassLoader
 *          负责加载核心类库
 *      2. 平台类加载器:PlatformClassLoader
 *          负责加载扩展类库
 *      3. 应用类加载器:AppClassLoader
 *          负责加载classpath
 *
 * 类加载器也是可以自定义的,只要符合类加载器的规范即可。
 * 自定义的类加载器,我们一般称为:用户类加载器。
 */
public class ReflectTest15 {
    public static void main(String[] args) {

        // 通过自定义的类获取的类加载器是:应用类加载器。
        ClassLoader appClassLoader = ReflectTest15.class.getClassLoader();
        System.out.println("应用类加载器:" + appClassLoader);

        // 获取应用类加载器
        ClassLoader appClassLoader2 = ClassLoader.getSystemClassLoader();
        System.out.println("应用类加载器:" + appClassLoader2);

        // 获取应用类加载器
        ClassLoader appClassLoader3 = Thread.currentThread().getContextClassLoader();
        System.out.println("应用类加载器:" + appClassLoader3);

        // 通过 getParent() 方法可以获取当前类加载器的 “父 类加载器”。
        // 获取平台类加载器。
        System.out.println("平台类加载器:" + appClassLoader.getParent());

        // 获取启动类加载器。
        // 注意:启动类加载器负责加载的是JDK核心类库,这个类加载器的名字看不到,直接输出的时候,结果是null。
        System.out.println("启动类加载器:" + appClassLoader.getParent().getParent());
    }
}

双亲委派机制

  1. 某个类加载器接收到加载类的任务时,通常委托给“父 类加载”完成加载。
  2. 最“父 类加载器”无法加载时,一级一级向下委托加载任务。
  3. 作用:
    1. 保护程序的安全。
    2. 防止类加载重复。

反射泛型

反射父类的泛型

/**
 * 获取父类的泛型信息
 */
public class Test {
    public static void main(String[] args) {
        // 获取类
        Class<Cat> catClass = Cat.class;

        // 获取当前类的父类泛型
        Type genericSuperclass = catClass.getGenericSuperclass();
        //System.out.println(genericSuperclass instanceof Class);
        //System.out.println(genericSuperclass instanceof ParameterizedType);

        // 如果父类使用了泛型
        if(genericSuperclass instanceof ParameterizedType){
            // 转型为参数化类型
            ParameterizedType parameterizedType = (ParameterizedType) genericSuperclass;
            // 获取泛型数组
            Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
            // 遍历泛型数组
            for(Type a : actualTypeArguments){
                // 获取泛型的具体类型名
                System.out.println(a.getTypeName());
            }
        }
    }
}

反射接口的泛型

Class<Mouse> mouseClass = Mouse.class;
        // 获取接口上的泛型
        Type[] genericInterfaces = mouseClass.getGenericInterfaces();
        for (Type g : genericInterfaces) {
            // 使用了泛型
            if(g instanceof ParameterizedType){
                ParameterizedType parameterizedType = (ParameterizedType) g;
                Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
                for(Type a : actualTypeArguments){
                    System.out.println(a.getTypeName());
                }
            }
        }

反射属性上的泛型

// 获取这个类
Class<User> userClass = User.class;
// 获取属性上的泛型,需要先获取到属性
Field mapField = userClass.getDeclaredField("map"); // 获取公开的以及私有的
// 获取这个属性上的泛型
Type genericType = mapField.getGenericType();
// 用泛型了
if(genericType instanceof ParameterizedType){
    ParameterizedType parameterizedType = (ParameterizedType) genericType;
    Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
    for(Type a : actualTypeArguments){
        System.out.println(a.getTypeName());
    }
}

反射方法参数上的泛型&方法返回值上的泛型

// 获取类
Class<MyClass> myClassClass = MyClass.class;

// 获取方法
Method mMethod = myClassClass.getDeclaredMethod("m", List.class, List.class);

// 获取方法参数上的泛型
Type[] genericParameterTypes = mMethod.getGenericParameterTypes();
for(Type g : genericParameterTypes){
    // 如果这个参数使用了泛型
    if(g instanceof ParameterizedType){
        ParameterizedType parameterizedType = (ParameterizedType) g;
        Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
        for(Type a : actualTypeArguments){
            System.out.println(a.getTypeName());
        }
    }
}

// 获取方法返回值上的泛型
Type genericReturnType = mMethod.getGenericReturnType();
if(genericReturnType instanceof ParameterizedType){
    ParameterizedType parameterizedType = (ParameterizedType) genericReturnType;
    Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
    for(Type a : actualTypeArguments){
        System.out.println(a.getTypeName());
    }
}

反射构造方法参数上的泛型

Class<User> userClass = User.class;
Constructor<User> con = userClass.getDeclaredConstructor(Map.class);
Type[] genericParameterTypes = con.getGenericParameterTypes();
for(Type g :genericParameterTypes){
    if(g instanceof ParameterizedType){
        ParameterizedType parameterizedType = (ParameterizedType) g;
        Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
        for(Type a : actualTypeArguments){
            System.out.println(a.getTypeName());
        }
    }
}
  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Java老狗

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值