关于JAVA泛型数组类型擦除引发的问题及解决方案
先看如下一个DEMO示例代码:(其中doBatchGet被子类重写了1次)
public abstract class BaseDemoService<T> {
public String batchGet(T... ints) {
T one=ints[0];
System.out.println("one class:" + one.getClass().getCanonicalName());
return doBatchGet(one);
}
public String doBatchGet(T... ints){
System.out.println("com.example.springwebapp.service.Demo1Service.doBatchGet");
return String.join(",", Arrays.toString(ints));
}
public String doGet(T one){
return one.toString();
}
}
@Service
public class Demo1Service extends BaseDemoService<Integer> {
@Override
public String doBatchGet(Integer... ints) {
System.out.println("com.example.springwebapp.service.Demo1Service.doBatchGet");
return super.doBatchGet(ints);
}
@Override
public String doGet(Integer one) {
return super.doGet(one);
}
}
执行单元测试,示例代码如下:
@Resource
private BaseDemoService baseDemoService;
@Test
public void testOverride(){
Integer[] ints={1,2,3,4,5,6};
System.out.println("batchGet result:" + baseDemoService.batchGet(ints));
}
大家认为最后batchGet result输出的结果如何?有人可能会认为这太简单了,就是1,但实际上的结果可能会让你打脸,居然直接报错,如下所示:
one class:java.lang.Integer
java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [Ljava.lang.Integer;
at com.example.springwebapp.service.Demo1Service.doBatchGet(Demo1Service.java:5)
at com.example.springwebapp.service.BaseDemoService.batchGet(BaseDemoService.java:11)
at com.example.springwebapp.Springwebapp1ApplicationTests.testOverride(Springwebapp1ApplicationTests.java:38)
... ...
报错的意思是类型转换错误,object数组类型不能转换为integer数组类型,但我上面一行明明输出的结果就是one class:java.lang.Integer,为何传给doBatchGet就变成了object类型呢?是不是有点看不懂了,编译也是正常的,但运行就是报错,如果把入参的ints直接传到doBatchGet方法入参中也是OK的,或者内部将doBatchGet改为doGet,即使子类重写doGet一样可以正确获取结果 又或者doBatchGet没有被子类重写也可以正确获得结果,但如果单独从ints中取一个成员然后再调doBatchGet方法就会报错,大家知道为啥吗?
我这里就不卖关子了,其实就是泛型参数类型擦除“惹的祸”,因为别看我们是BaseDemoService,其实这个T在编译后就是一个Object,在从ints取出一个成员值one后,虽然指明是T,但实际就是Object,最后在调doBatchGet的时候,JVM会将one这个变量隐式转换为one的Object数组,然后调doBatchGet,由于doBatchGet方法被子类重写,而重写后入参是具体的类型(Integer数组),这样就导致了上面的object数组类型不能转换为integer数组类型的错误。
当我们知道了问题原因后,那么就可以对症下药了.
解法一(关键点:通过反射创建一个入参真实类型的数组,这个数组与doBatchGet入参类型一致):
public String batchGet(T... ints) {
T[] arr = (T[]) Array.newInstance(ints[0].getClass(), 1);
arr[0]=ints[0];
System.out.println("arr class:" + arr.getClass().getCanonicalName());
return doBatchGet(arr);
}
解法二(关键点:将doBatchGet由原来的可变参数直接改为List,这样即使子类重写,因为List本身就是泛型,所以擦除了就都是一样的):
public String batchGet(T... ints) {
return doBatchGet(Arrays.asList(ints));
}
public String doBatchGet(List<T> ints){
System.out.println("com.example.springwebapp.service.Demo1Service.doBatchGet");
return ints.toString();
}
以上就是实际项目中碰到的问题及解法,希望对大家也有帮助。
关于JAVA泛型数组类型擦除引发的问题及解决方案的更多相关文章
- Java泛型:类型擦除
类型擦除 代码片段一 Class c1 = new ArrayList<Integer>().getClass(); Class c2 = new ArrayList<String& ...
- Java泛型之类型擦除
类型擦除 学过C++模板的,在使用Java泛型的时候,会感觉到有点不疑问,例如:(1)无法定义一个泛型数组.无法调用泛型参数对象中对应的方法(当然,通过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 ...
- Java泛型类与类型擦除
转载自:http://blog.csdn.net/lonelyroamer/article/details/7868820 一.Java泛型的实现方法:类型擦除 前面已经说了,Java的泛型是伪泛型. ...
- java之集合类框架的简要知识点:泛型的类型擦除
这里想说一下在集合框架前需要理解的小知识点,也是个人的肤浅理解,不知道理解的正不正确,请大家多多指教.这里必须谈一下java的泛型,因为它们联系紧密,我们先看一下这几行代码: Class c1 = n ...
随机推荐
- vue tabBar导航栏设计实现4-再次抽取MainTabBar
系列导航 一.vue tabBar导航栏设计实现1-初步设计 二.vue tabBar导航栏设计实现2-抽取tab-bar 三.vue tabBar导航栏设计实现3-进一步抽取tab-item 四.v ...
- Asp .Net Core 系列:集成 Ocelot+Nacos+Swagger+Cors实现网关、服务注册、服务发现
目录 简介 什么是 Ocelot ? 什么是 Nacos ? 什么是 Swagger ? 什么是 Cors ? Asp .Net Core 集成 Ocelot 网关集成 Nacos 下游配置 Naco ...
- vue后台管理系统,接口环境配置
https://coding.imooc.com/lesson/397.html#mid=31487
- 12_前K个高频元素
前K个高频元素 给你一个整数数组 nums 和一个整数 k ,请你返回其中出现频率前 k 高的元素.你可以按 任意顺序 返回答案.347.力扣题目链接 示例 1: 输入: nums = [1,1,1, ...
- 08-Shell计算命令
1.expr命令 expr (evaluate expressions 的缩写),译为"表达式求值".Shell expr 是一个功能强大,并且比较复杂的命令,它除了可以实现整数计 ...
- 【TouchGFX 】使用 CubeMX 创建 TouchGFX 工程时 LCD 死活不显示
生成的代码死活无法让LCD显示,经两个晚上的分析验证是LTDC_CLK引脚速度设置为低速导致,经测试中速.高速.超高速都正常,真是冤,聊以此以示纪念
- 单例bean与原型bean的区别
在使用Spring开发时,Spring提供了五种scope,分别为singleton,prototype,request,session,global session. 上图为各个scope描述的官方 ...
- 神经网络优化篇:详解调试处理(Tuning process)
调试处理 关于训练深度最难的事情之一是要处理的参数的数量,从学习速率\(a\)到Momentum(动量梯度下降法)的参数\(\beta\).如果使用Momentum或Adam优化算法的参数,\(\be ...
- [转帖]s3fs - 使用S3FS存储桶目录允许其他用户使用权限
https://www.coder.work/article/6661505 我在使用S3FS时遇到问题.我正在使用 ubuntu@ip-x-x-x-x:~$ /usr/bin/s3fs --ve ...
- [转帖]一本正经的八卦一下CPU的自主可控
https://zhuanlan.zhihu.com/p/36391482 (2018年的4月16日,美国商务部发布对中兴通讯出口权限禁令,禁止美国企业向其出售零部件,史称"中兴禁运事件&q ...