首页 >> 大全

Java泛型04 : 泛型类型擦除

2023-10-23 大全 28 作者:考证青年

超级通道: Java泛型学习系列-绪论

本章主要对Java泛型的类型擦除进行学习。

经过前面几个章节的学习,可以掌握泛型的基本用法,但是有可能这种掌握只是死记硬背。

为了从根本上掌握泛型的使用,我们需要正确的理解泛型,为了正确的理解泛型,我们需要首先理解类型擦除(Type )的概念。

1.类型擦除初识

先看Java泛型02 : 泛型原始类型、泛型类型命名规范、有界类型的一个例子。

/*** 

Title: 泛型原始类型使用

* @author 韩超 2018/2/22 11:21*/
public static void main(String[] args){//泛型原始类型MyGenericsType myGenericsType = new MyGenericsType();LOGGER.info(myGenericsType.getClass().toString());//泛型类型MyGenericsType integerMyGenericsType = new MyGenericsType();LOGGER.info(integerMyGenericsType.getClass().toString()); }

我们对这个例子进行修改,如下:

//类型擦除
LOGGER.info("类型擦除:");
MyGenericsType myGenericsType = new MyGenericsType();
MyGenericsType integerMyGenericsType = new MyGenericsType();
MyGenericsType doubleMyGenericsType = new MyGenericsType();
LOGGER.info(myGenericsType.getClass().toString());
LOGGER.info(integerMyGenericsType.getClass().toString());
LOGGER.info(doubleMyGenericsType.getClass().toString());

运行结果:

2018-02-23 09:41:53 INFO  TypeErasureDemo:22 - 类型擦除:
2018-02-23 09:41:53 INFO  TypeErasureDemo:26 - class pers.hanchao.generics.type.MyGenericsType
2018-02-23 09:41:53 INFO  TypeErasureDemo:27 - class pers.hanchao.generics.type.MyGenericsType
2018-02-23 09:41:53 INFO  TypeErasureDemo:28 - class pers.hanchao.generics.type.MyGenericsType

通过观察运行结果,可以发现,泛型在编译之后,只保留了原始类型,即。

泛型类型擦除:泛型在编译阶段,生成的字节码中不包含泛型类型,只保留原始类型。

2.类型擦除的原始类型

上面说到,类型擦除之后,只保留原始类型,那么原始类型到底是什么类型呢?

看下面一段代码:

/*** 

Title: 类型擦除之后的原始类型 示例

* @author 韩超 2018/2/23 10:32*/
static class TempList{private T t;public T setT(T t) {this.t = t;return t;} }public static void main(String[] args) {//类型擦除之后的原始类型System.out.println();LOGGER.info("泛型类型擦除之后的原始类型:");TempList tempList = new TempList();LOGGER.info(tempList.getClass().getDeclaredField("t").getType());Method[] methods = tempList.getClass().getDeclaredMethods();for (Method method : methods) {LOGGER.info(method.getReturnType() + " " + method.getName() + " (" + method.getParameterTypes()[0] + ")");} }

运行结果:

泛型和泛型擦除_泛型类型擦除_

2018-02-23 10:32:18 INFO  TypeErasureDemo:43 - 类型擦除之后的原始类型:
2018-02-23 10:32:18 INFO  TypeErasureDemo:45 - class java.lang.Object
2018-02-23 10:32:18 INFO  TypeErasureDemo:48 - class java.lang.Object setT (class java.lang.Object)

可以肯定:无界的泛型类型擦除之后的原始类型是类型。

上面的经过编译阶段的类型擦除,形成的原始类型如下:

static class TempList{private Object t;public Object setT(Object t) {this.t = t;return t;}
}

那么有界的泛型类型呢?

编写示例如下:

/*** 

Title: 有界泛型类型 类型擦除之后的原始类型 示例

* @author 韩超 2018/2/23 10:36*/
static class DemoList{private T t;public T setT(T t){this.t = t;return t;} } public static void main(String[] args) {//有界泛型类型擦除之后的原始类型System.out.println();LOGGER.info("有界泛型类型擦除之后的原始类型:");DemoList demoList = new DemoList();LOGGER.info(demoList.getClass().getDeclaredField("t").getType());Method[] methods = demoList.getClass().getDeclaredMethods();for (Method method : methods) {LOGGER.info(method.getReturnType() + " " + method.getName() + " (" + method.getParameterTypes()[0] + ")");}LOGGER.info("有界泛型类型擦除之后的原始类型:父类型"); }

运行结果:

2018-02-23 10:41:51 INFO  TypeErasureDemo:70 - 有界泛型类型擦除之后的原始类型:
2018-02-23 10:41:51 INFO  TypeErasureDemo:72 - class java.lang.Number
2018-02-23 10:41:51 INFO  TypeErasureDemo:75 - class java.lang.Number setT (class java.lang.Number)
2018-02-23 10:41:51 INFO  TypeErasureDemo:77 - 有界泛型类型擦除之后的原始类型:父类型

结论:有界泛型类型擦除之后的原始类型:父类型

那么多重有界的泛型类型呢?

编写示例如下:


/*** 

Title: 多重 有界泛型类型 类型擦除之后的原始类型 示例

* @author 韩超 2018/2/23 10:43*/
static class AList{private T t;public T setT(T t){this.t = t;return t;} } static class BList{private T t;public T setT(T t){this.t = t;return t;} } public static void main(String[] args){//多重有界泛型类型擦除之后的原始类型System.out.println();LOGGER.info("多重有界泛型类型擦除之后的原始类型:");AList aList = new AList();LOGGER.info(aList.getClass().getDeclaredField("t").getType());Method[] methods = aList.getClass().getDeclaredMethods();for (Method method : methods) {LOGGER.info(method.getReturnType() + " " + method.getName() + " (" + method.getParameterTypes()[0] + ")");}BList bList = new BList();LOGGER.info(bList.getClass().getDeclaredField("t").getType());Method[] methods2 = bList.getClass().getDeclaredMethods();for (Method method : methods2) {LOGGER.info(method.getReturnType() + " " + method.getName() + " (" + method.getParameterTypes()[0] + ")");}LOGGER.info("多重有界泛型类型擦除之后的原始类型:第一个父类型"); }

运行结果:

2018-02-23 10:47:38 INFO  TypeErasureDemo:100 - 多重有界泛型类型擦除之后的原始类型:
2018-02-23 10:47:38 INFO  TypeErasureDemo:102 - interface java.io.Serializable
2018-02-23 10:47:38 INFO  TypeErasureDemo:105 - interface java.io.Serializable setT (interface java.io.Serializable)
2018-02-23 10:47:38 INFO  TypeErasureDemo:108 - interface java.lang.Comparable
2018-02-23 10:47:38 INFO  TypeErasureDemo:111 - interface java.lang.Comparable setT (interface java.lang.Comparable)
2018-02-23 10:47:38 INFO  TypeErasureDemo:113 - 多重有界泛型类型擦除之后的原始类型:第一个父类型

结论:多重有界泛型类型擦除之后的原始类型:第一个父类型

3.先检查后编译

既然泛型会在编译阶段进行类型擦除,那如何保证编译之前的类型转换安全呢?答案是:通过IDE提供的编译前检查功能

泛型和泛型擦除_泛型类型擦除_

再看下面一段代码:

//类型擦除与编译前检查
LOGGER.info("类型擦除与编译前检查:");
MyGenericsType integerMyGenericsType1 = new MyGenericsType();
//类型检查通过
integerMyGenericsType1.setT(new Integer(1));
//类型检查不通过,在IDE中报错
integerMyGenericsType1.setT(new Double(2D));

上述代码在IDE( ,集成开发环境)中,无法通过编译前检查,会直接报错,如下:

这里写图片描述

总结:Java泛型先检查在编译

4.通过反射跳过类型检查

通过上面的章节已知,无法将类型的数据写入到类型的泛型中,那么久真的无法实现吗?

其实是可以的:通过Java反射可以跳过泛型的类型检查。

//通过反射跳过类型检查
LOGGER.info("通过反射跳过类型检查:");
MyGenericsType doubleMyGenericsType1 = new MyGenericsType();
//类型检查通过
doubleMyGenericsType1.setT(new Double(2D));
//类型检查不通过
//doubleMyGenericsType.setT(new Integer(1));
//通过反射跳过类型检查
doubleMyGenericsType1.getClass().getMethod("setT",Object.class).invoke(doubleMyGenericsType1,new Integer(1));
LOGGER.info(doubleMyGenericsType1.getT());

2018-02-23 09:57:40 INFO  TypeErasureDemo:41 - 通过反射跳过类型检查:
2018-02-23 09:57:40 INFO  TypeErasureDemo:49 - 1

上面的示例,通过反射将类型的数据写入了类型的泛型对象中,并且能够get出来,类型为。

那么实际上1存储的是什么类型呢?

我们运行如下代码:

LOGGER.info(doubleMyGenericsType1.getT().getClass().toString());

会报错:

Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Doubleat pers.hanchao.generics.erasure.TypeErasureDemo.main(TypeErasureDemo.java:50)

通过报错分析,实际1实际存储的是类型。

tags: 泛型

关于我们

最火推荐

小编推荐

联系我们


版权声明:本站内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 88@qq.com 举报,一经查实,本站将立刻删除。备案号:桂ICP备2021009421号
Powered By Z-BlogPHP.
复制成功
微信号:
我知道了