Java泛型之类型未被擦除
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泛型之类型未被擦除的更多相关文章
- Java泛型:类型擦除
类型擦除 代码片段一 Class c1 = new ArrayList<Integer>().getClass(); Class c2 = new ArrayList<String& ...
- Java泛型之类型擦除
类型擦除 学过C++模板的,在使用Java泛型的时候,会感觉到有点不疑问,例如:(1)无法定义一个泛型数组.无法调用泛型参数对象中对应的方法(当然,通过extends关键字是可以做到,只是比较麻烦): ...
- Java 泛型 通配符类型
Java 泛型 通配符类型 @author ixenos 摘要:限定通配符类型.无限定通配符类型.与普通泛型区别.通配符捕获 通配符类型 通配符的子类型限定(?都是儿孙) <? extends ...
- java 泛型的类型擦除和桥方法
oracle原文地址:https://docs.oracle.com/javase/tutorial/java/generics/erasure.html 在Java中,泛型的引入是为了在编译时提供强 ...
- Java泛型-类型擦除
一.概述 Java泛型在使用过程有诸多的问题,如不存在List<String>.class, List<Integer>不能赋值给List<Number>(不可协变 ...
- 转:有关Java泛型的类型擦除(type erasing)
转载自:拈花微笑 自从Java 5引入泛型之后,Java与C++对于泛型不同的实现的优劣便一直是饭后的谈资.在我之前的很多training中,当讲到Java泛型时总是会和C++的实现比较,一般得出的结 ...
- java 泛型的类型擦除与桥方法
泛型类 --代码参考:java核心技术 卷1 第十版 public class Pair<T> { private T first; private T second; //构造器 pub ...
- JAVA 泛型之类型擦除
★ 泛型是 JDK 1.5 版本引进的概念,之前是没有泛型的概念的,但泛型代码能够很好地和之前版本的代码很好地兼容. CollectionTest.java ---编译成CollectionTest. ...
- Java泛型的类型擦除
package com.srie.testjava; import java.util.ArrayList; import java.util.List; public class TestGener ...
随机推荐
- 安装node-sass的正确姿势【转】
安装 node-sass 的时候总是会各种不成功,今天我琢磨了一会儿总算知道要怎么解决了. 首先要知道的是,安装 node-sass 时在 node scripts/install 阶段会从 gith ...
- perl6正则 2: 字母,数字,空格,下划线, 字符集
数字, 字母, 下划线 在perl6中, 如果是 数字, 字母, 下划线, 在正则里可以正接写上. > so / True > so 'perl6_' ~~ /_/ True > 非 ...
- juery下拉刷新,div加载更多元素并添加点击事件(二)
buffer.append("<div class='col-xs-3 "+companyId+"' style='padding-left: 10px; padd ...
- yolo回归型的物体检测
本弱又搬了另外一个博客的讲解: 缩进YOLO全称You Only Look Once: Unified, Real-Time Object Detection,是在CVPR2016提出的一种目标检测算 ...
- caffe源码整个训练过程
Caffe源码 Blob protected: shared_ptr<SyncedMemory> data_; shared_ptr<SyncedMemory> diff_; ...
- windows下安装多个mysql
1.正常安装mysql5.1.33 安装服务名为mysql3306 安装目录d:\mysql5.1\3306 安装完成后,关闭服务 ① 复制安装文件 将默认安装目录C:\Documents and S ...
- Linux环境Nginx安装、调试以及PHP安装
linux版本:64位CentOS 6.4 Nginx版本:nginx1.8.0 php版本:php5.5 1.编译安装Nginx 官网:http://wiki.nginx.org/Install 下 ...
- python之pandas&&DataFrame(二)
简单操作 Python-层次聚类-Hierarchical clustering >>> data = pd.Series(np.random.randn(10),index=[[' ...
- Fiddler Web Session 列表(1)
Web Session 列表 位置: Web Session 列表 位于Fiddler界面的左侧 ,是Fiddler所抓取到的所有Session会话的列表集合. Web Session 列表 栏名词解 ...
- PHP任意文件上传漏洞CVE-2015-2348浅析
昨晚安全新闻爆出一个“PHP任意文件上传漏洞”,CVE编号为:CVE-2015-2348. 当时楼主正准备收拾东西回家,看到这个新闻心里一惊:失传江湖多年的0字符截断上传漏洞又重现了?而且还影响这么多 ...