1. 泛型擦除

泛型是在 Jdk1.5 之后引入的,为了使字节码向前兼容,Java编译器会在编译时擦除泛型信息(泛型擦除)。假使有泛型类Xx<T>,对其进行反射并打印类中的泛型方法:

public class Xx<T> {
public void m(T t) {}
public void m(T[] arr) {}
public void m(List<T> list) {}
}
Method[] declaredMethods = Xx.class.getDeclaredMethods();
for (Method declaredMethod : declaredMethods) {
System.out.println(declaredMethod);
}

打印信息如下:

public void Xx.m(java.util.List)
public void Xx.m(java.lang.Object[])
public void Xx.m(java.lang.Object)

2. 桥接方法

假使有类Xx2继承自Xx<String>,则显然Xx2并未覆盖Xx中的两个方法:

1. public void Xx.m(java.lang.Object)

2. public void Xx.m(java.lang.Object[])

public class Xx2 extends Xx<String> {
@Override
public void m(String t) {}
@Override
public void m(String[] arr) {}
@Override
public void m(List<String> list) {}
}

对Xx2进行反射并打印类中的方法:

Method[] declaredMethods = Xx2.class.getDeclaredMethods();
for (Method declaredMethod : declaredMethods) {
System.out.println(declaredMethod);
}
public void Xx2.m(java.util.List)
public void Xx2.m(java.lang.Object[])
public void Xx2.m(java.lang.Object)
public void Xx2.m(java.lang.String[])
public void Xx2.m(java.lang.String)

可见Xx2源码被编译成字节码后,额外生成了两个方法:

1. public void Xx2.m(java.lang.Object)

2. public void Xx2.m(java.lang.Object[])

这两个方法被称为桥接方法。桥接方法覆盖父类中的泛型方法,并在内部调用子类中的具体方法:

public void m(Object t) {
m((String) t);
}
public void m(Object[] arr) {
m((String[]) arr);
}

可以使用Method.isBridge来判断方法是否为桥接方法:

Method[] declaredMethods = Xx2.class.getDeclaredMethods();
for (Method declaredMethod : declaredMethods) {
System.out.println(declaredMethod);
System.out.println(" -isBridge: " + declaredMethod.isBridge());
}
public void Xx2.m(java.util.List)
-isBridge: false
public void Xx2.m(java.lang.Object[])
-isBridge: true
public void Xx2.m(java.lang.Object)
-isBridge: true
public void Xx2.m(java.lang.String[])
-isBridge: false
public void Xx2.m(java.lang.String)
-isBridge: false

3. 获取泛型信息

Java编译器会擦除方法内部的泛型信息:

void m() {
  List<String> list = new ArrayList<>();
}

因此,方法内部全部对象的泛型信息将丢失。但Java编译器也在字节码中保留三处泛型信息:

1. 类定义的泛型信息

2. 类中字段的泛型信息

3. 方法参数的泛型信息

Java泛型信息由java.lang.reflect.Type的五个子类进行描述:

1. java.lang.Class(如:java.lang.String)

2. java.lang.reflect.TypeVariable(如:T)

3. java.lang.reflect.WildcardType(如:? extends Comparable)

4. java.lang.reflect.GenericArrayType(如:T[])

5. java.lang.reflect.ParameterizedType(如:List<String>)

3.1 获取类定义上的泛型信息

1. Class.getTypeParameters

TypeVariable<Class<Xx>> typeParameter = Xx.class.getTypeParameters()[0];
System.out.println(typeParameter + ": " + typeParameter.getClass());

打印信息如下:

T: class sun.reflect.generics.reflectiveObjects.TypeVariableImpl

2. Class.getGenericSuperClass / Class.getGenericInterfaces

Type genericSuperclass = Xx2.class.getGenericSuperclass();
System.out.println(genericSuperclass + ": " + genericSuperclass.getClass());

打印信息如下:

Xx<java.lang.String>: class sun.reflect.generics.reflectiveObjects.ParameterizedTypeImpl

3.2 类中字段的泛型信息(Field.getGenericType)

List<? extends String> list;
Field list = Xx.class.getDeclaredField("list");
Type genericType = list.getGenericType();
System.out.println(genericType + ": " + genericType.getClass());

打印信息如下:

java.util.List<? extends java.lang.String>: class sun.reflect.generics.reflectiveObjects.ParameterizedTypeImpl

3.3 方法参数的泛型信息(Method.getGenericParameterTypes / Method.getGenericReturnType)

Method m = Xx.class.getDeclaredMethod("m", Object[].class);
Type parameterType = m.getGenericParameterTypes()[0];
System.out.println(parameterType + ": " + parameterType.getClass());

打印信息如下:

T[]: class sun.reflect.generics.reflectiveObjects.GenericArrayTypeImpl

Java泛型之类型未被擦除的更多相关文章

  1. Java泛型:类型擦除

    类型擦除 代码片段一 Class c1 = new ArrayList<Integer>().getClass(); Class c2 = new ArrayList<String& ...

  2. Java泛型之类型擦除

    类型擦除 学过C++模板的,在使用Java泛型的时候,会感觉到有点不疑问,例如:(1)无法定义一个泛型数组.无法调用泛型参数对象中对应的方法(当然,通过extends关键字是可以做到,只是比较麻烦): ...

  3. Java 泛型 通配符类型

    Java 泛型 通配符类型 @author ixenos 摘要:限定通配符类型.无限定通配符类型.与普通泛型区别.通配符捕获 通配符类型 通配符的子类型限定(?都是儿孙) <? extends ...

  4. java 泛型的类型擦除和桥方法

    oracle原文地址:https://docs.oracle.com/javase/tutorial/java/generics/erasure.html 在Java中,泛型的引入是为了在编译时提供强 ...

  5. Java泛型-类型擦除

    一.概述 Java泛型在使用过程有诸多的问题,如不存在List<String>.class, List<Integer>不能赋值给List<Number>(不可协变 ...

  6. 转:有关Java泛型的类型擦除(type erasing)

    转载自:拈花微笑 自从Java 5引入泛型之后,Java与C++对于泛型不同的实现的优劣便一直是饭后的谈资.在我之前的很多training中,当讲到Java泛型时总是会和C++的实现比较,一般得出的结 ...

  7. java 泛型的类型擦除与桥方法

    泛型类 --代码参考:java核心技术 卷1 第十版 public class Pair<T> { private T first; private T second; //构造器 pub ...

  8. JAVA 泛型之类型擦除

    ★ 泛型是 JDK 1.5 版本引进的概念,之前是没有泛型的概念的,但泛型代码能够很好地和之前版本的代码很好地兼容. CollectionTest.java ---编译成CollectionTest. ...

  9. Java泛型的类型擦除

    package com.srie.testjava; import java.util.ArrayList; import java.util.List; public class TestGener ...

随机推荐

  1. Feather包实现数据框快速读写,你值得拥有

    什么是Feather? Feature是一种文件格式,支持R语言和Python的交互式存储,速度更快.目前支持R语言的data.frame和Python pandas 的DataFrame. Feat ...

  2. java中的matches -> 完全匹配

    matches是完全匹配.跟matcher不一样, matcher像perl正则, 能匹配到符合的都会返回true, 而这个matches要完全一模一样才行. import java.util.reg ...

  3. MySQL常见错误代码说明

    附:MySQL常见错误代码说明 1005:创建表失败 1006:创建数据库失败 1007:数据库已存在,创建数据库失败 1008:数据库不存在,删除数据库失败 1009:不能删除数据库文件导致删除数据 ...

  4. Redis 启动警告解决【转】

    [root@centos224]# service redisd start :M Nov :: (it was originally set to ). _._ _.-``__ ''-._ _.-` ...

  5. SVN的使用、分支合并及解决冲突详解

    一.什么是SVN SVN是Subversion的简称,是一个开放源代码的版本控制系统,相较于RCS.CVS,它采用了分支管理系统,它的设计目标就是取代CVS. 二.SVN的下载安装 下载地址:http ...

  6. java经典面试题大全

    基本概念 操作系统中 heap 和 stack 的区别 什么是基于注解的切面实现 什么是 对象/关系 映射集成模块 什么是 Java 的反射机制 什么是 ACID BS与CS的联系与区别 Cookie ...

  7. wondows下安装pytho&pip

    1.在https://www.python.org/downloads/下载相应的python安装包, 解压安装,配置环境变量. 2.下载pip安装包:https://pypi.python.org/ ...

  8. Java学习(API及Object类、String类、StringBuffer字符串缓冲区)

    一.JAVA的API及Object类 1.API 概念: Java 的API(API: Application(应用) Programming(程序) Interface(接口)) Java API就 ...

  9. pymongo的一些操作

    参考:http://www.yiibai.com/mongodb/mongodb_drop_collection.html http://www.cnblogs.com/zhouxuchen/p/55 ...

  10. System.Net.Mail的应用,后端发送邮件

    private void btn_send_Click(object sender, EventArgs e) { var emailAcount = ConfigurationManager.App ...