type java_Java中的Type

下面就用代码的方式,对其中的5大类型:原始类型(Class)、参数化类型(ParameterizedType)、数组类型(GenericArrayType)、类型变量(TypeVariable)、基本类型(Class) 进一步说明;

1.ParameterizedType

ParameterizedType表示参数化类型,也就是泛型,例如List、Set等;

b7e28f97b15f9dc4e1dbcdc82281845f.png

ParameterizedType

在ParameterizedType接口中,有3个方法,分别是getActualTypeArguments()、 getRawType()、 getOwnerType();

1.1 getActualTypeArguments

获取泛型中的实际类型,可能会存在多个泛型,例如Map,所以会返回Type[]数组;

2b2f4e886b9dc63f137e52ffae5ef434.png

值得注意的是,无论<>中有几层嵌套(List),getActualTypeArguments()方法永远都是脱去最外层的<>(也就是List<>),将口号内的内容(Map)返回;

我们经常遇到的List,通过getActualTypeArguments()方法,得到的返回值是TypeVariableImpl对象,也就是TypeVariable类型(后面介绍);

1.2 getRawType

获取声明泛型的类或者接口,也就是泛型中<>前面的那个值;

0111f899f577c02047d4c604a65d88c3.png

1.3 getOwnerType

通过方法的名称,我们大概了解到,此方法是获取泛型的拥有者,那么拥有者是个什么意思?

Returns a {@code Type} object representing the type that this type    * is a member of.  For example, if this type is {@code O.I},    * return a representation of {@code O}.  (摘自JDK注释)

通过注解,我们得知,“拥有者”表示的含义--内部类的“父类”,通过getOwnerType()方法可以获取到内部类的“拥有者”;例如: Map  就是 Map.Entry的拥有者;

152c041e30cd7c98a10d2ade23abf0fc.png

2.GenericArrayType

泛型数组类型,例如List[] 、T[]等;

4448d75cdba339c27e5983633fd23549.png

GenericArrayType

在GenericArrayType接口中,仅有1个方法,就是getGenericComponentType();

2.1 getGenericComponentType

返回泛型数组中元素的Type类型,即List[] 中的 List(ParameterizedTypeImpl)、T[] 中的T(TypeVariableImpl);

5fde34937b5342101c4818350e221294.png

值得注意的是,无论是几维数组,getGenericComponentType()方法都只会脱去最右边的[],返回剩下的值;

3.TypeVariable

泛型的类型变量,指的是List、Map中的T,K,V等值,实际的Java类型是TypeVariableImpl(TypeVariable的子类);此外,还可以对类型变量加上extend限定,这样会有类型变量对应的上限;

961444c33acb2526ffcdb6f4f086ba01.png

TypeVariable

在TypeVariable接口中,有3个方法,分别为getBounds()、getGenericDeclaration()、getName();

3.1 getBounds

获得该类型变量的上限,也就是泛型中extend右边的值;例如 List ,Number就是类型变量T的上限;如果我们只是简单的声明了List(无显式定义extends),那么默认为Object;

29fc996f9721e6ece20b9570efa8ca70.png

无显式定义extends:

3e84a71a3d8b088fd55df321ac2c3e01.png

值得注意的是,类型变量的上限可以为多个,必须使用&符号相连接,例如 List;其中,& 后必须为接口;

3.2 getGenericDeclaration

获取声明该类型变量实体,也就是TypeVariableTest中的TypeVariableTest;

674ff1a8e60404501f56906faa36efdf.png

3.3 getName

获取类型变量在源码中定义的名称;

6cd3de8ec22133548ff0f74cd3664b06.png

说到TypeVariable类,就不得不提及Java-Type体系中另一个比较重要的接口---GenericDeclaration;含义为:声明类型变量的所有实体的公共接口;也就是说该接口定义了哪些地方可以定义类型变量(泛型);

通过查看源码发现,GenericDeclaration下有三个子类,分别为Class、Method、Constructor;也就是说,我们定义泛型只能在一个类中这3个地方自定义泛型;

3dfd6196b7acf45adcc42e4d110b382e.png

此时,我们不禁要问,我们不是经常在类中的属性声明泛型吗,怎么Field没有实现 GenericDeclaration接口呢?

其实,我们在Field中并没有声明泛型,而是在使用泛型而已!不信,我们实际上代码来看看!

1.首先在Class上定义泛型:

2f097452d7f2a17e5b5b61109c71aa04.png

Class定义泛型

2.我们没有在Class上定义泛型,直接在构造方法上定义泛型

b9b05934a010c0bbbb6a0af08a9fdab3.png

泛型构造

3.同样没有在Class定义泛型,直接在普通方法上定义泛型

e1c474b9fa26d5f937a5fbbc24edaaff.png

泛型方法

3.我们直接在属性上定义

6cfa88d97526d3855af94f3c319733d9.png

属性上定义泛型

我们看到,如果不在Class上定义,属性上并不能直接使用!所以,这也是我之前说的属性上并不是定义泛型,而是使用泛型,所以Field并没有实现GenericDeclaration接口!

4.Class

Type接口的实现类,是我们工作中常用到的一个对象;在Java中,每个.class文件在程序运行期间,都对应着一个Class对象,这个对象保存有这个类的全部信息;因此,Class对象也称之为Java反射的基础;

2c97674c0d75a1b0031b178cd716d126.png

Class

通过上面的例子,可以看出,当我们没有声明泛型的时候,我们普通的对象就是一个Class类型,是Type中的一种;

5.WildcardType

?---通配符表达式,表示通配符泛型,但是WildcardType并不属于Java-Type中的一钟;例如:List extends Number> 和 List super Integer>;

d6558bc83b5d32257b019bf8a5b6e33b.png

WildcardType

在WildcardType接口中,有2个方法,分别为getUpperBounds()、getLowerBounds();

5.1 getUpperBounds

获取泛型变量的上边界(extends)

d7f9ce5f3859f40e5e7f74fb8b90c114.png

5.2 getLowerBounds

获取泛型变量的下边界(super)

15900413a37bed7f26fbd3c5d9fcc538.png

以上,就是对Java-Type体系中相关对象的介绍;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值