关于泛型擦除问题的解决--Mybatis查询类型转换

 更新时间:2022年08月25日 14:22:53   作者:keep丶  
这篇文章主要介绍了关于泛型擦除问题的解决--Mybatis查询类型转换方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

概念介绍

Java语言的泛型采用的是擦除法实现的伪泛型,泛型信息(类型变量、参数化类型)编译之后通通被除掉了。使用擦除法的好处就是实现简单、非常容易Backport,运行期也能够节省一些类型所占的内存空间。

而擦除法的坏处就是,通过这种机制实现的泛型远不如真泛型灵活和强大。Java选取这种方法是一种折中,因为Java最开始的版本是不支持泛型的,为了兼容以前的库而不得不使用擦除法。

验证擦除,我们编写下面代码:

public class ErasedTypeEquivalence {
    public static void main(String[] args) {
    	//例1
        ArrayList<String> list1 = new ArrayList<String>();
        list1.add("abc");
        ArrayList<Integer> list2 = new ArrayList<Integer>();
        list2.add(123);
        System.out.println(list1.getClass() == list2.getClass());//true
		
		//例2
		ArrayList<Integer> list = new ArrayList<Integer>();
        list.add(1);  //这样调用 add 方法只能存储整形,因为泛型类型的实例为 Integer
        list.getClass().getMethod("add", Object.class).invoke(list, "asd");
        for (int i = 0; i < list.size(); i++) {
            System.out.println(list.get(i));//会输出1和asd
        }
    }
}

在例1中,我们定义了两个ArrayList数组,不过一个是ArrayList<String>泛型类型的,只能存储字符串;一个是ArrayList<Integer>泛型类型的,只能存储整数,最后,我们通过list1对象和list2对象的getClass()方法获取他们的类的信息,最后发现结果为true。说明泛型类型String和Integer都被擦除掉了,只剩下原始类型。       

在例2中,定义了一个ArrayList泛型类型实例化为Integer对象,如果直接调用add()方法,那么只能存储整数数据,不过当我们利用反射调用add()方法的时候,却可以存储字符串,这说明了Integer泛型实例在编译之后被擦除掉了,只保留了原始类型。

上面两次提到了原始类型,什么是原始类型?原始类型 就是擦除去了泛型信息,最后在字节码中的类型变量的真正类型,无论何时定义一个泛型,相应的原始类型都会被自动提供,类型变量擦除,并使用其限定类型(无限定的变量用Object)替换。

问题案例

最近在搭系统基础代码架构,其中就涉及到系统数据字典 功能,以前都是用varchar类型保存字典内容,这次准备玩点新花样,准备用上MySQL的JSON类型保存字典表的内容字段。>>文章传送门<<

实际操作之后就遇到了泛型擦除问题,如下图,我虽然对content字段的List指定了泛型DictContent,但是在做类型转换时,只能指定javaType=List,没有也不能指定其泛型:

在没有指定泛型的情况下,JacksonTypeHandler在做类型转换后生成的集合的泛型就与预期的不一致:

原因分析

原因很简单,在resultMap中指定的JavaType是java.util.List,此处只能指定类类型,并不能指定泛型。而在对应的类型转换类中也没有指定其泛型,而List<DictContent>和List<Object>的类类型是一样的,所以在给content字段赋值时是不会报错的。但是一旦你需要操作List的中的元素,在取出元素时,JVM就发现你要的类型是DictContent 而实际上是LinkedHashMap,就会抛出类型转换异常。

通俗的讲就是你准备买华为手机(将JSON类型转成List<DictContent>类型),但是买的时候没有说要买什么牌子的手机(在javaType中只指定了List类型,没有也无法指定泛型),而店子里有很多牌子的手机,所以店家就随便给了你一款手机。。。

以下是Mybatis Plus中的部分源码,可以看到在没有指定List的泛型的情况下,通过JacksonTypeHandler处理后的元素类型并不是我们预期的类型:

下图我们可以看到JacksonTypeHandler是BaseTypeHandler的子类,而且指定了BaseTypeHandler中的泛型是Object类型,但是上图中的泛型却是LinkedHashMap。

至于为什么是LinkedHashMap,我觉得是JVM指定的,如果哪位大佬比较清楚这块的逻辑还请在评论中指点一下!

解决方案

既然原因搞清楚了,解决方案就呼之欲出了,有两种方案:

  • 自定义一个指定泛型的集合类替代List<T>
  • 引用上文中通俗的说法,这个方案就是在买手机的时候告诉卖家,我要买华为手机。
  • 自定义一个指定泛型的TypeHandler类替代JacksonTypeHandler类
  • 而这里的的通俗的说法就是让店家只卖华为手机。

以上两种方案都可以实现我们的需求。

从工作量上来说,自定义一个List<T>显然更少,所以我选择了第一种方案,如图:

8.11新增:第二种解决方式:

替换后结果如下:

至此,泛型擦除问题解决。 

总结

不得不说,玩新花样总是会遇到各种各样的坑,但是编程之路,不就是不断的踩坑,不断的改BUG,积累经验,打怪升级。如果不是因为最近玩了这个新花样,可能我这辈子都不会遇到泛型擦除的问题!       

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • java使用selenium自动化WebDriver等待的示例代码

    java使用selenium自动化WebDriver等待的示例代码

    显式等待和隐式等待是WebDriver中两种常用的等待方式,它们都可以用来等待特定的条件满足后再继续执行代码,本文给大家介绍java使用selenium自动化WebDriver等待,感兴趣的朋友一起看看吧
    2023-09-09
  • 详解Java8新特性如何防止空指针异常

    详解Java8新特性如何防止空指针异常

    要说 Java 编程中哪个异常是你印象最深刻的,那 NullPointerException 空指针可以说是臭名昭著的,不要说初级程序员会碰到, 即使是中级,专家级程序员稍不留神,就会掉入这个坑里,本文就和大家聊聊Java8新特性如何防止空指针异常
    2023-08-08
  • java中的GC收集器详情

    java中的GC收集器详情

    这篇文章主要介绍了java中的GC收集器,GC(Garbage collection )指的是程序内存管理分手动和自动,手动内存管理,需要我们编程的时候显式分配和释放空间,但如果忘记释放,会造成严重的内存泄漏问题,下面文章内容我们就来实例说明情况,需要的朋友可以参考一下
    2021-10-10
  • SpringCache缓存自定义配置的实现

    SpringCache缓存自定义配置的实现

    本文主要介绍了SpringCache缓存自定义配置的实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01
  • Eclipse中实现JS代码提示功能(图文教程)

    Eclipse中实现JS代码提示功能(图文教程)

    本文通过图文并茂的形式给大家介绍了Eclipse中实现JS代码提示功能,非常不错,具有参考借鉴价值,需要的朋友参考下吧
    2017-11-11
  • SSM项目实现短信验证码登录功能的示例代码

    SSM项目实现短信验证码登录功能的示例代码

    这篇文章主要为大家分享了在SSM项目中实现短信验证码登录功能的示例代码,文中的实现步骤讲解详细,感兴趣的小伙伴可以跟随小编一起动手尝试一下
    2022-05-05
  • 浅聊一下Spring中Bean的配置细节

    浅聊一下Spring中Bean的配置细节

    我们知道,当写完一个普通的 Java 类后,想让 Spring IoC 容器在创建类的实例对象时使用构造方法完成实例对象的依赖注入,那么就需要在配置元数据中写好类的 Bean 定义,包括各种标签的属性。所以本文我们来说说这其中的配置细节,需要的朋友可以参考下
    2023-07-07
  • Java编译器用maven打war包出错解决办法

    Java编译器用maven打war包出错解决办法

    这篇文章主要介绍了用maven打war包出错的解决办法,需要的朋友可以参考下
    2018-03-03
  • Netty分布式客户端接入流程初始化源码分析

    Netty分布式客户端接入流程初始化源码分析

    这篇文章主要介绍了Netty分布式客户端接入流程初始化源码分析,有关channelConfig有关的初始化过程剖析,有需要的朋友可以借鉴参考下,希望能够有所帮助
    2022-03-03
  • springboot全局字符编码设置方式(解决乱码问题)

    springboot全局字符编码设置方式(解决乱码问题)

    这篇文章主要介绍了springboot全局字符编码设置方式(解决乱码问题),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12

最新评论