Type是Java 编程语言中所有类型的公共高级接口(官方解释),也就是Java中所有类型的“爹”;其中,“所有类型”的描述尤为值得关注。它并不是我们平常工作中经常使用的 int、String、List、Map等数据类型,而是从Java语言角度来说,对基本类型、引用类型向上的抽象;

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

原始类型,不仅仅包含我们平常所指的类,还包括枚举、数组、注解等;

参数化类型,就是我们平常所用到的泛型List、Map;

数组类型,并不是我们工作中所使用的数组String[] 、byte[],而是带有泛型的数组,即T[] ;

基本类型,也就是我们所说的java的基本类型,即int,float,double等

Type的直接子接口

  • ParameterizedType: 表示一种参数化的类型,比如Collection,即普通的泛型。

ParameterizedType 是参数化类型,即泛型,类似List、Map<Integer, String>、List<? extends Number>带有类型参数的类型,也可以是自定义的,再调用getRawType()与getActualTypeArguments()两个方法,就可以得到声明此参数化类型的类(java.lang.Comparable)和实际的类型参数数组([? super T]),而这个? super T又是一个WildcardType类型。

public interface ParameterizedType extends Type {
    // 获取泛型的类型,比如 Map<Integer,String> 获取的是 Integer和String
    Type[] getActualTypeArguments();

    // 获取<> 前面的类型,比如 Map<Integer,String> 获取的是Map
    Type getRawType();

     // For example, if this type is {@code O<T>.I<S>}, return a representation of {@code O<T>}.
    Type getOwnerType();
}

示例

public class Test {
    public static void main(String[] args) {

        Field[] declaredFields = Person.class.getDeclaredFields();
        for(Field f:declaredFields){
            if(f.getGenericType() instanceof ParameterizedType){
                ParameterizedType genericType = (ParameterizedType) f.getGenericType();
                System.out.println(String.format("%s 属性的原生类型为%s,参数类型有",f.getName(),genericType.getRawType()));
                for(Type type : genericType.getActualTypeArguments()){
                    System.out.println(type);
                }
                System.out.println("================");
            }
        }
    }
}

class Person {

    List<Integer> integerList;
    Map<String,Integer> belongs;

}

// 输出

属性 integerList 的原生类型为interface java.util.List,参数类型有
class java.lang.Integer
================
属性 belongs 的原生类型为interface java.util.Map,参数类型有
class java.lang.String
class java.lang.Integer
================
  • TypeVariable:是各种类型变量的公共父接口,就是泛型里面的类似T、E。 ** 在这需要强调的是,TypeVariable代表着泛型中的变量,而ParameterizedType则代表整个泛型; **

认识TypeVariable之前,我们先来看另一个接口 GenericDecalaration

public interface GenericDeclaration extends AnnotatedElement {
    /**
     * 返回的泛型变量数组代表了泛型声明的内容,TypeVariable类型就是定义泛型变量的。
     */
    public TypeVariable<?>[] getTypeParameters();
}

GenericDeclaration 有三个直接子类 Class,Construtor,Method,也就是说只能在这几种对象上进行范型变量的声明(定义)。GenericDeclaration的接口方法getTypeParameters用来逐个获取该GenericDeclaration的范型变量声明。

//1.在类(Class)上声明(定义)类型变量
class A<T>{
  T a;
}//之后这里可用任意类型替换T,例如
A<String> as = new A<String>();
//是否看着有点像集合?不错,集合就是泛型的一个典型运用

//2.在方法上声明(定义,方法上,类型变量声明(定义)不是在参数里边,而且必须在返回值之前,static等修饰后
public <E> void test(E e){}


//3.声明(定义)在构造器上
public <K> A(K k){}

【注意】类型变量声明(定义)的时候不能有下限(既不能有super),否则编译报错。为什么?T extends classA表示泛型有上限classA,当然可以,因为这样,每一个传进来的类型必定是classA(具有classA的一切属性和方法),但若是T super classA,传进来的类型不一定具有classA的属性和方法,当然就不适用于泛型

[extends 和 super的区别]

关键字说明
  ? 通配符类型
 <? extends T> 表示类型的上界,表示参数化类型的可能是T 或是 T的子类
<? super T> 表示类型下界(Java Core中叫超类型限定),表示参数化类型是此类型的超类型(父类型),直至Object
看了这个我是不太明白,换成白话是这个意思:
List<? extends T> 是说 这个list放的是T或者T的子类型的对象,但是不能确定具体是什么类型,所以可以get(),不能add()(可以add null值)

List<? super T> 是说这个list放的是至少是T类型的对象,所以我可以add T或者T的子类型,但是get得到的类型不确定,所以不能获取
public class TestSuperExtends {
    public static void main(String[] args) {
        // 造数
        List<Apple> apples = new ArrayList<>();
        apples.add(new Apple());
        apples.add(new Apple());
        apples.add(new Apple());

        ArrayList<Food> foods = new ArrayList<>();
        ArrayList<RedApple> redApples = new ArrayList<>();


   
        // 任何Food的子类的列表都能赋值成功,也可以取值,但是不能添加除null以外的元素
        List<? extends Food> foodList = apples;
//        foodList.add(new Apple()); // error
//        foodList.add(new Food());  // error
        foodList.add(null);
        test(foods);

        
        //  因为super表示下届,所以foods和apples都可以赋值成功,但是Apple的子类不行。
        List<? super Apple> list1 = foods;
        List<? super Apple> list = apples;
//        List<? super Apple> list2 = redApples;  // error


        // 可以添加Apple及其子类,但是不是添加任何Apple的父类
//        list.add(new Food()); // error
        list.add(new Apple());
        list.add(new RedApple());
        Object object = list.get(0);

    }

    public static <T extends Food> void  test(List<T> foods){
        for(T f:foods){
            f.eat();
        }
    }




}

我们知道Object类有一个方法, getClass(), 它返回当前实例的运行时类, 而Class是 GenericDeclaration 的子类,所以 Java所有的类都能通过调用 getTypeParameters() 方法获取到类型变量.

TypeVariable是类型变量,如参数化类型中的E、K等类型变量,表示泛指任何类,如果加上extends/super限定,则就会有相应的上限、下限。

public interface TypeVariable<D extends GenericDeclaration> extends Type, AnnotatedElement {
    /**
     * 获得该类型变量的上限(上边界),若无显式定义(extends),默认为Object,类型变量的上限可能不止一个,因为可以用&符号限定多个(这其中有且只能有一个为类或抽象类,且必须放在extends后的第一个,即若有多个上边界,则第一个&后必为接口)
     * 比如 xxxClass<K extends List & Serialize>,则返回 List和Serialize
    */
    Type[] getBounds();

    /**
     *  获取的是在哪个类上进行的泛型声明,在如上的例子中,返回的是xxxClass
     * @since 1.5
     */
    D getGenericDeclaration();

    /**
     *  同样是上面的例子,返回 K, 也就是泛型类型声明时写的名字
     */
    String getName();

    /**
     * 
     * @since 1.8
     */
     AnnotatedType[] getAnnotatedBounds();
}
public class Test<K extends List & Serializable, V> {
    public static void main(String[] args) {
        TypeVariable[] v = new Test().getClass().getTypeParameters();

        for(TypeVariable t:v){
            System.out.println(Arrays.stream(t.getBounds()).map(Type::getTypeName).collect(Collectors.toList()));
            System.out.println(t.getName());
            System.out.println(t.getGenericDeclaration());
            System.out.println("-----------------");
        }

    }
}

打印输出如下

[java.util.List, java.io.Serializable]
K
class com.lion.converter.test.Test
-----------------
[java.lang.Object]
V
class com.lion.converter.test.Test
  • GenericArrayType:表示一种元素类型是参数化类型或者类型变量的数组类型,比如List<>[],T[]这种。
public interface GenericArrayType extends Type {
    /**
     *  获取泛型数组中元素的类型,要注意的是:
  无论从左向右有几个[]并列,这个方法仅仅脱去最右边的[]之后剩下的内容就作为这个方法的返回值。
     */
    Type getGenericComponentType();
}
  • WildcardType:代表一种通配符类型表达式,类似? super T这样的通配符表达式。

WildcardType是通配符表达式,或泛型表达式,它虽然是Type的一个子接口,但并不是Java类型中的一种,表示的仅仅是类似 ? extends T、? super K这样的通配符表达式。

public interface WildcardType extends Type {
    /**
     * 获取泛型表达式上界
     */
    Type[] getUpperBounds();

    /**
     * 获取泛型表达式下界
     */
    Type[] getLowerBounds();
   
}