【Spring源码分析】ResolvableType

【Spring源码分析】ResolvableType

参考
目录

文章目录

      • 【Spring源码分析】ResolvableType
          • 一、ParameterizedType 参数化类型,即泛型;例如:List< T>、Map< K,V>等带有参数化的对象;
          • 二、GenericArrayType—— 泛型数组 泛型数组,描述的是形如:A< T>[]或T[]类型变量和原始类型
          • 三、TypeVariable
          • 四、WildcardType
          • 五、获取父接口父类中的泛型参数,使用原生和spring ResolvableType

Type
在这里插入图片描述

所有类型的类型是常见的超接口Java编程语言。这些包括原始类型,参数化的类型,数组类型,类型变量和原始类型(These include raw types, parameterized types, array types, type variables and primitive types).它并不是我们平常工作中经常使用的 int、String、List、Map等数据类型,而是从Java语言角度来说,对基本类型、引用类型向上的抽象;

Type体系中类型的包括:原始类型(Class)、参数化类型(ParameterizedType)、数组类型(GenericArrayType)、类型变量(TypeVariable)、基本类型(Class);

原始类型,不仅仅包含我们平常所指的类,还包括枚举、数组、注解等;
参数化类型,就是我们平常所用到的泛型List、Map;
数组类型,并不是我们工作中所使用的数组String[] 、byte[],而是带有泛型的数组,即T[] ;
基本类型,也就是我们所说的java的基本类型,即int,float,double等

一、ParameterizedType 参数化类型,即泛型;例如:List< T>、Map< K,V>等带有参数化的对象;

作用于Class的Field上面

Type[] getActualTypeArguments(); 返回 这个 Type 类型的参数的实际类型数组。

package com.lvyuanj.standard.resolvabletypes;

import lombok.extern.slf4j.Slf4j;

import java.lang.reflect.Field;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Map;
import java.util.Set;

/**
 * ParameterizedType 使用测试 参数化类型
 *
 */
@Slf4j
public class ParameterizedTypeTest {

    /**
     * 1、map: 获取ParameterizedType:class java.lang.String
     * 2、map: 获取ParameterizedType:class com.wangji.demo.ParameterizedTypeTest
     * 3、map:getOwnerType is null
     * 4、map:getRawType:interface java.util.Map
     */
    private Map<String, ParameterizedTypeTest> map;
    /**
     * 1、set1: 获取ParameterizedType:class java.lang.String
     * 2、set1:getOwnerType is null
     * 3、set1:getRawType:interface java.util.Set
     */
    private Set<String> set1;
    /**
     * 1、 clz: 获取ParameterizedType:?
     * 2、 clz:getOwnerType is null
     * 3、clz:getRawType:class java.lang.Class
     */
    private Class<?> clz;
    /**
     * 1、holder: 获取ParameterizedType:class java.lang.String
     * 2、holder:getOwnerType:class com.wangji.demo.ParameterizedTypeTest
     * 3、holder:getRawType:class com.wangji.demo.ParameterizedTypeTest$Holder
     */
    private Holder<String> holder;

    /**
     * 1、list: 获取ParameterizedType:class java.lang.String
     * 2、list:getOwnerType is null
     * 3、list:getRawType:interface java.util.List
     */
    private List<String> list;
    /**
     * str:is not ParameterizedType
     */
    private String str;
    /**
     * i:is not ParameterizedType
     */
    private Integer i;
    /**
     * set:is not ParameterizedType
     */
    private Set set;
    /**
     *  aList:is not ParameterizedType
     */
    private List aList;
    /**
     * 1、entry: 获取ParameterizedType:class java.lang.String
     * 2、entry: 获取ParameterizedType:class java.lang.String
     * 3、entry:getOwnerType:interface java.util.Map
     * 4、entry:getRawType:interface java.util.Map$Entry
     */
    private Map.Entry<String, String> entry;


    static class Holder<V> {

    }

    public static void testParameterizedType() {
        Field f = null;
        try {
            Field[] fields = ParameterizedTypeTest.class.getDeclaredFields();
            for (int i = 0; i < fields.length; i++) {
                f = fields[i];
                if(f.getName().equals("log")){
                    continue;
                }
                System.out.println(f.getName()+" == " + f.getType());
                if(f.getGenericType() instanceof ParameterizedType){
                    ParameterizedType parameterizedType = (ParameterizedType) f.getGenericType();
                    for(Type type :parameterizedType.getActualTypeArguments()){
                        log.info(f.getName()+": 获取ParameterizedType:"+type);
                    }
                    if(parameterizedType.getOwnerType() !=null){
                        log.info(f.getName()+":getOwnerType:"+parameterizedType.getOwnerType());
                    }else{
                        log.info(f.getName()+":getOwnerType is null");
                    }
                    if(parameterizedType.getRawType() !=null){
                        log.info(f.getName()+":getRawType:"+parameterizedType.getRawType());
                    }
                }else{
                    log.info(f.getName() + ":is not ParameterizedType ");
                }
            }
        }catch (Exception e){
            log.error("error",e);
        }
    }


    public static void main(String[] args) {
        testParameterizedType();
    }
}

运行结果如下:

map == interface java.util.Map
11:04:52.041 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - map: 获取ParameterizedType:class java.lang.String
11:04:52.047 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - map: 获取ParameterizedType:class com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest
11:04:52.047 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - map:getOwnerType is null
11:04:52.048 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - map:getRawType:interface java.util.Map
set1 == interface java.util.Set
11:04:52.049 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - set1: 获取ParameterizedType:class java.lang.String
11:04:52.049 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - set1:getOwnerType is null
11:04:52.049 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - set1:getRawType:interface java.util.Set
clz == class java.lang.Class
11:04:52.050 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - clz: 获取ParameterizedType:?
11:04:52.050 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - clz:getOwnerType is null
11:04:52.050 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - clz:getRawType:class java.lang.Class
holder == class com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest$Holder
11:04:52.051 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - holder: 获取ParameterizedType:class java.lang.String
11:04:52.051 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - holder:getOwnerType:class com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest
11:04:52.051 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - holder:getRawType:class com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest$Holder
list == interface java.util.List
11:04:52.051 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - list: 获取ParameterizedType:class java.lang.String
11:04:52.051 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - list:getOwnerType is null
11:04:52.052 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - list:getRawType:interface java.util.List
str == class java.lang.String
11:04:52.052 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - str:is not ParameterizedType 
i == class java.lang.Integer
11:04:52.052 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - i:is not ParameterizedType 
set == interface java.util.Set
11:04:52.052 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - set:is not ParameterizedType 
aList == interface java.util.List
11:04:52.052 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - aList:is not ParameterizedType 
entry == interface java.util.Map$Entry
11:04:52.053 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - entry: 获取ParameterizedType:class java.lang.String
11:04:52.053 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - entry: 获取ParameterizedType:class java.lang.String
11:04:52.053 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - entry:getOwnerType:interface java.util.Map
11:04:52.053 [main] INFO  com.lvyuanj.standard.resolvabletypes.ParameterizedTypeTest - entry:getRawType:interface java.util.Map$Entry
二、GenericArrayType—— 泛型数组 泛型数组,描述的是形如:A< T>[]或T[]类型变量和原始类型

作用于Class的method上面

package com.lvyuanj.standard.resolvabletypes;

import lombok.extern.slf4j.Slf4j;

import java.lang.reflect.*;
import java.util.List;

/**
 * GenericArrayType—— 泛型数组
 * 泛型数组,描述的是形如:A<T>[]或T[]类型
 *   is either a parameterized type or a type variable.
 */
@Slf4j
public class GenericArrayTypeTest<T> {

    /**
     * 含有泛型数组的才是GenericArrayType
     * @param pTypeArray GenericArrayType type :java.util.List<java.lang.String>[];
     * genericComponentType:java.util.List<java.lang.String>
     * @param vTypeArray  GenericArrayType type :T[];genericComponentType:T
     * @param list ParameterizedType type :java.util.List<java.lang.String>;
     * @param strings type :class [Ljava.lang.String;
     * @param test type :class [Lcom.wangji.demo.GenericArrayTypeTest;
     */
    public void testGenericArrayType(List<String>[] pTypeArray, T[] vTypeArray
    , List<String> list, String[] strings, GenericArrayTypeTest[] test) {
    }

    /**
     * 1、getGenericComponentType
     * 返回泛型数组中元素的Type类型,即List<String>[] 中的 List<String>(ParameterizedTypeImpl)
     * 、T[] 中的T(TypeVariableImpl);
     * 值得注意的是,无论是几维数组,getGenericComponentType()方法都只会脱去最右边的[],返回剩下的值;
     */
    public static void testGenericArrayType() {
        Method[] declaredMethods = GenericArrayTypeTest.class.getDeclaredMethods();
        for(Method method :declaredMethods){
            if(method.getName().startsWith("main")){
                continue;
            }
            log.info("declare Method:"+method);
            /**
             * 获取当前参数所有的类型信息
             */
            Type[] types = method.getGenericParameterTypes();
            for(Type type: types){
                if(type instanceof ParameterizedType){
                    log.info("ParameterizedType type :"+type);
                }else if(type instanceof GenericArrayType){
                    log.info("GenericArrayType type :"+type);
                    Type genericComponentType = ((GenericArrayType) type).getGenericComponentType();
                    /**
                     * 获取泛型数组中元素的类型,要注意的是:无论从左向右有几个[]并列,
                     * 这个方法仅仅脱去最右边的[]之后剩下的内容就作为这个方法的返回值。
                     */
                    log.info("genericComponentType:"+genericComponentType);
                }else if(type instanceof WildcardType){
                    log.info("WildcardType type :"+type);
                }else if(type instanceof  TypeVariable){
                    log.info("TypeVariable type :"+type);
                }else {
                    log.info("type :"+type);
                }
            }
        }
    }

    public static void main(String[] args) {
        testGenericArrayType();
    }
}

运行结果:

11:31:23.706 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - declare Method:public void com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest.testGenericArrayType(java.util.List[],java.lang.Object[],java.util.List,java.lang.String[],com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest[])
11:31:23.718 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - GenericArrayType type :java.util.List<java.lang.String>[]
11:31:23.718 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - genericComponentType:java.util.List<java.lang.String>
11:31:23.718 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - GenericArrayType type :T[]
11:31:23.719 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - genericComponentType:T
11:31:23.719 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - ParameterizedType type :java.util.List<java.lang.String>
11:31:23.719 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - type :class [Ljava.lang.String;
11:31:23.719 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - type :class [Lcom.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest;
11:31:23.719 [main] INFO  com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest - declare Method:public static void com.lvyuanj.standard.resolvabletypes.GenericArrayTypeTest.testGenericArrayType()
三、TypeVariable

TypeVariable Interface TypeVariable< D extends GenericDeclaration>
在这里插入图片描述

泛型的类型变量,指的是List< T>、Map< K,V>中的T,K,V等值,实际的Java类型是TypeVariableImpl(TypeVariable的子类;此外,还可以对类型变量加上extend限定,这样会有类型变量对应的上限;值得注意的是,类型变量的上限可以为多个,必须使用&符号相连接,例如 List< T extends Number & Serializable>;其中,& 后必须为接口;

1、Type[] getBounds() 类型对应的上限,默认为Object
2、D getGenericDeclaration() 获取声明该类型变量实体,也就是TypeVariableTest< T>中的TypeVariableTest
3、String getName() 获取类型变量在源码中定义的名称;

package com.lvyuanj.standard.resolvabletypes;

import lombok.extern.slf4j.Slf4j;

import java.io.Serializable;
import java.lang.reflect.*;
import java.util.List;

/**
 * Interface TypeVariable<D extends GenericDeclaration> , 
 * D - the type of generic declaration that declared the underlying type variable.
 * 类型变量是类型变量的公共超接口。类型变量是第一次使用反射方法创建的,如在这个包中指定的。
 * 如果类型变量T由类型(即类、接口或注释类型)T引用,并且T由第n个封闭类T(参见JLS.1.2)来声明,
 * 那么T的创建需要T的第i个包围类的分辨率(参见JVMS 5),对于i=0到n,包含。创建类型变量不能导致其边界的创建。 
 * 重复创建类型变量没有任何效果。
 * <p>
 * 可以在运行时实例化多个对象以表示给定的类型变量。即使类型变量只创建一次, 
 * 但这并不意味着缓存表示类型变量的实例的任何要求。
 * 但是,表示一个类型变量的所有实例必须是相等的()。因此,类型变量的用户不能依赖实现该接口的类实例的标识。
 * <p>
 * 泛型的类型变量,指的是List<T>、Map<K,V>中的T,K,V等值,实际的Java类型是TypeVariableImpl 
 * (TypeVariable的子类); 
 * 此外,还可以对类型变量加上extend限定,这样会有类型变量对应的上限;值得注意的是,类型变量的上限可以为多个, 
 * 必须使用&符号相连接,例如 List<T extends Number & Serializable>;其中,& 后必须为接口;
 * 没有指定的话 ,V 的 上边界 属于  Object
 */
@Slf4j
public class TypeVariableTest<T extends Number & Serializable, V> {
    /**
     * TypeVariable
     */
    private T key;

    /**
     * TypeVariable
     */
    private V value;

    /**
     * GenericArrayType V[]-> V TypeVariable 两种混合起来了
     */
    private V[] values;
    /**
     * 原始类型,不仅仅包含我们平常所指的类,还包括枚举、数组、注解等;
     * 基本类型,也就是我们所说的java的基本类型,即int,float,double等
     */
    private String str;

    /**
     * 获取ParameterizedType List<T> -> T TypeVariable 两种混合起来了
     */
    private List<T> tList;

    /**
     * 从这个例子中可以看出来各种类型之间是相互在使用的
     * TypeVariable<D extends GenericDeclaration>
     * GenericDeclaration  All Known Implementing Classes: Class, Constructor, Method

     */
    public static void testTypeVariableTest() {
        Field f = null;
        try {
            Field[] fields = TypeVariableTest.class.getDeclaredFields();
            for (int i = 0; i < fields.length; i++) {
                f = fields[i];
                if (f.getName().equals("log")) {
                    continue;
                }
                log.info("begin ******当前field:" + f.getName() + " *************************");
                if (f.getGenericType() instanceof ParameterizedType) {
                    ParameterizedType parameterizedType = (ParameterizedType) f.getGenericType();
                    for (Type type : parameterizedType.getActualTypeArguments()) {
                        log.info(f.getName() + ": 获取ParameterizedType:" + type);
                        if (type instanceof TypeVariable) {
                            printTypeVariable(f.getName(), (TypeVariable) type);
                        }
                    }
                    if (parameterizedType.getOwnerType() != null) {
                        log.info(f.getName() + ":getOwnerType:" + parameterizedType.getOwnerType());
                    } else {
                        log.info(f.getName() + ":getOwnerType is null");
                    }
                    if (parameterizedType.getRawType() != null) {
                        log.info(f.getName() + ":getRawType:" + parameterizedType.getRawType());
                    }
                } else if (f.getGenericType() instanceof GenericArrayType) {
                    GenericArrayType genericArrayType = (GenericArrayType) f.getGenericType();
                    log.info("GenericArrayType type :" + genericArrayType);
                    Type genericComponentType = genericArrayType.getGenericComponentType();
                    if (genericComponentType instanceof TypeVariable) {
                        TypeVariable typeVariable = (TypeVariable) genericComponentType;
                        printTypeVariable(f.getName(), typeVariable);
                    }
                } else if (f.getGenericType() instanceof TypeVariable) {
                    TypeVariable typeVariable = (TypeVariable) f.getGenericType();
                    printTypeVariable(f.getName(), typeVariable);
                } else {
                    log.info("type :" + f.getGenericType());
                }
                log.info("end ******当前field:" + f.getName() + " *************************");
            }
        } catch (Exception e) {
            log.error("error", e);
        }
    }

    /**
     * 1、Type[] getBounds() 类型对应的上限,默认为Object
     * 2、D getGenericDeclaration()  获取声明该类型变量实体,也就是TypeVariableTest<T>中的TypeVariableTest
     * 3、String getName() 获取类型变量在源码中定义的名称;
     *
     * @param fieldName
     * @param typeVariable
     */
    private static void printTypeVariable(String fieldName, TypeVariable typeVariable) {
        for (Type type : typeVariable.getBounds()) {
            log.info(fieldName + ": TypeVariable getBounds " + type);
        }
        log.info("定义Class getGenericDeclaration: " + typeVariable.getGenericDeclaration());
        log.info("getName: " + typeVariable.getName());
    }

    public static void main(String[] args) {
        testTypeVariableTest();
    }

}

运行结果

11:39:48.697 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - begin ******当前field:key *************************
11:39:48.708 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - key: TypeVariable getBounds class java.lang.Number
11:39:48.709 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - key: TypeVariable getBounds interface java.io.Serializable
11:39:48.711 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - 定义Class getGenericDeclaration: class com.lvyuanj.standard.resolvabletypes.TypeVariableTest
11:39:48.711 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - getName: T
11:39:48.711 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - end ******当前field:key *************************
11:39:48.711 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - begin ******当前field:value *************************
11:39:48.711 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - value: TypeVariable getBounds class java.lang.Object
11:39:48.711 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - 定义Class getGenericDeclaration: class com.lvyuanj.standard.resolvabletypes.TypeVariableTest
11:39:48.712 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - getName: V
11:39:48.712 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - end ******当前field:value *************************
11:39:48.712 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - begin ******当前field:values *************************
11:39:48.713 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - GenericArrayType type :V[]
11:39:48.713 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - values: TypeVariable getBounds class java.lang.Object
11:39:48.713 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - 定义Class getGenericDeclaration: class com.lvyuanj.standard.resolvabletypes.TypeVariableTest
11:39:48.713 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - getName: V
11:39:48.713 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - end ******当前field:values *************************
11:39:48.713 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - begin ******当前field:str *************************
11:39:48.713 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - type :class java.lang.String
11:39:48.714 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - end ******当前field:str *************************
11:39:48.716 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - begin ******当前field:tList *************************
11:39:48.717 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - tList: 获取ParameterizedType:T
11:39:48.717 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - tList: TypeVariable getBounds class java.lang.Number
11:39:48.717 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - tList: TypeVariable getBounds interface java.io.Serializable
11:39:48.717 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - 定义Class getGenericDeclaration: class com.lvyuanj.standard.resolvabletypes.TypeVariableTest
11:39:48.717 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - getName: T
11:39:48.718 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - tList:getOwnerType is null
11:39:48.718 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - tList:getRawType:interface java.util.List
11:39:48.718 [main] INFO  com.lvyuanj.standard.resolvabletypes.TypeVariableTest - end ******当前field:tList *************************
四、WildcardType

WildcardType represents a wildcard type expression, such as ?, ? extends Number, or ? super Integer.
通配符表达式,或泛型表达式,它虽然是Type的一个子接口,但并不是Java类型中的一种,表示的仅仅是类似 ? extends T、? super K这样的通配符表达式。 ?—通配符表达式,表示通配符泛型,但是WildcardType并不属于Java-Type中的一钟;例如:List< ? extends Number> 和 List< ? super Integer>;
1、Type[] getUpperBounds(); //获得泛型表达式上界(上限) 获取泛型变量的上边界(extends)
2、Type[] getLowerBounds(); //获得泛型表达式下界(下限) 获取泛型变量的下边界(super)

package com.lvyuanj.standard.resolvabletypes;

import lombok.extern.slf4j.Slf4j;

import java.lang.reflect.*;
import java.util.List;

/**
 * WildcardType represents a wildcard type expression, such as ?, ? extends Number,  
 * or ? super Integer.
 * 通配符表达式,或泛型表达式,它虽然是Type的一个子接口,但并不是Java类型中的一种,表示的仅仅是类似 
 *  ? extends T、? super K这样的通配符表达式。
 * ?---通配符表达式,表示通配符泛型,但是WildcardType并不属于Java-Type中的一钟; 
 * 例如:List< ? extends Number> 和 List< ? super Integer>;
 * 1、Type[] getUpperBounds();  //获得泛型表达式上界(上限) 获取泛型变量的上边界(extends)
 * 2、Type[] getLowerBounds(); //获得泛型表达式下界(下限) 获取泛型变量的下边界(super)
 *
 */
@Slf4j
public class WildcardTypeTest {
    /**
     * 1、 a: 获取ParameterizedType:? extends java.lang.Number
     * 2、上界:class java.lang.Number
     */
    private List< ? extends Number> a;

    /**
     * b: 获取ParameterizedType:? super java.lang.String
     *  上届:class java.lang.Object
     *  下届:class java.lang.String
     */
    private List< ? super String> b;

    /**
     * c: 获取ParameterizedType:class java.lang.String
     */
    private List<String> c;

    /**
     * aClass: 获取ParameterizedType:?
     * 上届:class java.lang.Object
     */
    private Class<?> aClass;

    private String wangji;

    /**
     * 多种数据进行混合
     */
    public static void testWildcardType() {
        Field f = null;
        try {
            Field[] fields = WildcardTypeTest.class.getDeclaredFields();
            for (int i = 0; i < fields.length; i++) {
                f = fields[i];
                if (f.getName().equals("log")) {
                    continue;
                }
                log.info("begin ******当前field:" + f.getName() + " *************************");
                if (f.getGenericType() instanceof ParameterizedType) {
                    ParameterizedType parameterizedType = (ParameterizedType) f.getGenericType();
                    for (Type type : parameterizedType.getActualTypeArguments()) {
                        log.info(f.getName() + ": 获取ParameterizedType:" + type);
                        if (type instanceof WildcardType) {
                            printWildcardType((WildcardType) type);
                        }
                    }
                } else if (f.getGenericType() instanceof GenericArrayType) {
                    GenericArrayType genericArrayType = (GenericArrayType) f.getGenericType();
                    log.info("GenericArrayType type :" + genericArrayType);
                    Type genericComponentType = genericArrayType.getGenericComponentType();
                    if (genericComponentType instanceof WildcardType) {
                        printWildcardType((WildcardType) genericComponentType);
                    }
                } else if (f.getGenericType() instanceof TypeVariable) {
                    TypeVariable typeVariable = (TypeVariable) f.getGenericType();
                    log.info("typeVariable:" + typeVariable);

                } else {
                    log.info("type :" + f.getGenericType());
                    if (f.getGenericType() instanceof WildcardType) {
                        printWildcardType((WildcardType) f.getGenericType());
                    }
                }
                log.info("end ******当前field:" + f.getName() + " *************************");
            }
        } catch (Exception e) {
            log.error("error", e);
        }
    }

    private static void printWildcardType(WildcardType wildcardType) {
        for (Type type : wildcardType.getUpperBounds()) {
            log.info("上界:" + type);
        }
        for (Type type : wildcardType.getLowerBounds()) {
            log.info("下界:" + type);
        }
    }

    public static void main(String[] args) {
        testWildcardType();
    }
}

运行结果:

11:48:56.258 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - begin ******当前field:a *************************
11:48:56.287 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - a: 获取ParameterizedType:? extends java.lang.Number
11:48:56.287 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - 上界:class java.lang.Number
11:48:56.287 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - end ******当前field:a *************************
11:48:56.288 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - begin ******当前field:b *************************
11:48:56.288 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - b: 获取ParameterizedType:? super java.lang.String
11:48:56.288 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - 上界:class java.lang.Object
11:48:56.288 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - 下界:class java.lang.String
11:48:56.288 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - end ******当前field:b *************************
11:48:56.289 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - begin ******当前field:c *************************
11:48:56.289 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - c: 获取ParameterizedType:class java.lang.String
11:48:56.289 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - end ******当前field:c *************************
11:48:56.289 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - begin ******当前field:aClass *************************
11:48:56.290 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - aClass: 获取ParameterizedType:?
11:48:56.290 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - 上界:class java.lang.Object
11:48:56.291 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - end ******当前field:aClass *************************
11:48:56.291 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - begin ******当前field:wangji *************************
11:48:56.292 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - type :class java.lang.String
11:48:56.292 [main] INFO  com.lvyuanj.standard.resolvabletypes.WildcardTypeTest - end ******当前field:wangji *************************
五、获取父接口父类中的泛型参数,使用原生和spring ResolvableType
public class Parent<T> {
}


public interface IParent<T> {
}

使用原生和spring ResolvableType的比较

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/609400.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

竖排文字识别原理与实践操作方法

在当今数字化时代&#xff0c;OCR&#xff08;Optical Character Recognition&#xff0c;光学字符识别&#xff09;技术已经广泛应用于各个领域&#xff0c;特别是在文档处理方面&#xff0c;OCR软件能够帮助用户快速将纸质文档转化为可编辑的电子文档。然而&#xff0c;对于竖…

day-31 给植物浇水 II

思路 用两个变量start和end记录浇水位置&#xff0c;当前剩余水量大于需要浇水量时&#xff0c;进行浇水并前进一步&#xff0c;否则需要返回加水&#xff08;即重新灌满水罐的次数1&#xff09; 解题方法 用while&#xff08;start<end&#xff09;进行上述循环&#xff0…

【Spark】Spark编程体验,RDD转换算子、执行算子操作(六)

Spark编程体验 项目依赖管理 <dependencies><dependency><groupId>org.scala-lang</groupId><artifactId>scala-library</artifactId><version>2.12.10</version></dependency><dependency><groupId>org.ap…

数据库干货:推荐一款非常好用的 SQL Server管理工具

目录 2.1 SQL 编码辅助 2.2 表设计器 2.3 数据库设计器 2.4 模式比较 2.5 文档生成工具 2.6 数据导出和数据导入 2.7 源代码控制 2.8 监控工具 2.9 索引管理器 2.10 T-SQL 调试器 2.11 单元测试 一、软件简介 dbForge Studio 2019-2022 for SQL Server是针对SQL Serv…

Codeforces Round 943 (Div. 3)----B题题解

本题是很显然的双指针算法&#xff0c;一直移动&#xff0c;直达不匹配为止 #include <iostream> #include <vector> #include <string> #include <algorithm> using namespace std;const int N 200010; int t,n,m; char a[N],b[N];int main(void) {…

【国产SSL】哪家SSL证书可以保证数据不出境,是在国内验签

随着网络安全的重视&#xff0c;网站安装SSL证书已经是标配了。但是为什么目前常见的SSL证书都是国外的&#xff1f;数据受国外掌控&#xff0c;安全吗&#xff1f;那么哪家国产品牌是可以保证数据不出境的呢&#xff1f; 为什么目前常见的SSL证书都是国外的&#xff1f; 原因…

AI图书推荐:给自媒体创作者的ChatGPT使用指南

你是否厌倦了花费数小时盯着空白屏幕&#xff0c;努力为你的内容想出新鲜点子&#xff1f;想要将你的写作提升到下一个水平&#xff1f;有了ChatGPT&#xff0c;你可以告别写作障碍、无休止的修订和浪费的时间。 在这本全面的指南中&#xff0c;你将学到关于ChatGPT你需要知道…

BigInteger和BigDecimal类

BigInteger 和 BigDecimal 介绍 应用场景 BigInteger适合保存比较大的整型BigDecimal适合保存精度更高的浮点型&#xff08;小数&#xff09; BigInteger 和 BigDecimal 常见方法 1&#xff0c;add 加2&#xff0c;subtract 减3&#xff0c;multiply 乘4&#xff0c;divide…

2024年人工智能威胁态势报告:有关AI系统及AI应用的安全风险与安全防护全景

HiddenLayer公司最新发布的《2024年AI威胁场景报告》中&#xff0c;研究人员阐明了AI相关漏洞及其对组织的影响&#xff0c;并为应对这些挑战的IT安全和数据科学领导者提供了指导建议。最后&#xff0c;报告还揭示了各种形式的AI安全控制的前沿进展。 关键数据 平均而言&#x…

1. 介绍 Matplotlib

目录 1. Matplotlib 介绍 2. Matplotlib 安装介绍 3. Matplotlib 使用介绍 本专栏想分享一下最近学到的深度学习知识&#xff0c;前期需要一些 Python 知识和数据分析知识&#xff0c;如果有同学没有了解&#xff0c;请先看下面两个专栏&#xff0c;谢谢&#xff01; Pytho…

Ubuntu 网卡启动及配置

文章目录 问题分析查看网卡信息启动网卡 配置网卡应用更改 参考 问题分析 打开虚拟机后发现没有网卡网络。 查看网卡信息 sudo ip link set ens33 up得到本机的所有网卡信息&#xff0c;例如我这边网卡为ens33 启动网卡 启动网卡后发现依然网卡没有IP地址。 配置网卡 u…

【异常检测】新版异常检测库anomalib的使用

every blog every motto: There’s only one corner of the universe you can be sure of improving, and that’s your own self. https://blog.csdn.net/weixin_39190382?spm1010.2135.3001.5343 0. 前言 异常检测库anomalib的使用 1. 前提 1.1 数据组织形式 说明&#…

畅享清凉乐趣,气膜水上乐园的五大优点—轻空间

夏日的到来&#xff0c;让我们一同探索气膜水上乐园所带来的独特魅力。这里有着不同于传统水上乐园的五大优点&#xff0c;让您尽情享受清凉乐趣&#xff0c;感受畅游的无限畅快&#xff01; 1. 全年无休&#xff0c;尽享水上乐趣 气膜水上乐园的特殊建造材料和创新设计&#x…

qml 调用 C++函数

1、新建一个类 1)继承自QObject类&#xff08;记得添加QObject头文件&#xff09; 2)增加Q_OBJECT宏 3&#xff09;在需要调用的类成员函数前添加Q_INVOKEABLE声明 注意1&#xff1a;如果没有2&#xff09;和3&#xff09;&#xff0c;运行后提示函数类型错误 注意2&#…

CTF数据安全大赛crypto题目解题过程

CTF-Crypto加密题目内容 下面是一个Base64加密的密文 bXNobnszODdoajM3MzM1NzExMzQxMmo4MGg0bDVoMDYzNDQzNH0原文链接&#xff1a; 数据安全大赛CTF-Crypto题目 - 红客网-网络安全与渗透技术 我们用Python写一个解密脚本&#xff1a; import base64 import time #base64加密…

外贸人绝不能错过的欧洲市场如何开发?

欧洲市场作为世界上消费较大且较高的市场之一&#xff0c;容量大且接纳性强&#xff0c;不少外贸企业都希望在欧洲市场上掘金。 今天&#xff0c;我们就一起来看外贸人绝不能错过的欧洲市场如何开发&#xff1f;现在的欧洲还有哪些机会&#xff1f; 一、欧洲基本情况及市场特点…

各种依赖注入和分层解耦

分层解耦 三层架构 controller:控制层&#xff0c;接收前端发送的请求&#xff0c;对请求进行处理&#xff0c;并响应数据 service:业务逻辑层&#xff0c;处理具体业务的逻辑 dao:数据访问&#xff0c;负责数据访问操作&#xff0c;包括数据的增、删、改、查 流程为&…

wsl2安装rancher并导入和创建k8s集群

环境准备 安装wsl2点击此文]ubuntu20.04安装docker 点击此文,安装完成后docker镜像仓库改成阿里云镜像加速地址.如果不熟请点击此文 docker 安装rancher 启动wsl,根据官方文档以root身份执行 sudo docker run -d --restartunless-stopped -p 80:80 -p 443:443 --privileged …

位图转矢量图在线Ai神器,让你的图片无限放大不失真的秘密武器,重点免费、免费、免费

&#x1f4bb;开局附上神器地址&#xff1a;Convert PNG, JPG files to SVG vectors online - svg.tmttool.com 这边结合自己的使用场景简单做下使用教程&#x1f447;商业场景中经常需要对Ai生成的插画图进行二次创作&#xff0c;了解Ai做图的小伙伴就知道生成的图分辨率有限&…

2W 3KVDC 隔离 稳压单输出 DC/DC 电源模块——TPB 系列

TPB系列的输出带稳压和短路保护&#xff0c;特别适合于对输出电压精度和输出纹波噪声有要求的场合&#xff0c;完全兼容TPR和TPG的安装要求。