使用 SpringBoot2 + spring-data-redis 做缓存 ,GenericFastJsonRedisSerializer 做序列化和反序列化的情况下,devtools 开启本地热部署后,导致异常出现:

java.lang.ClassCastException: DTOObject cannot be cast to DTOObject

某个类不能强制转换成某个类,但是这两个类确实是一模一样的,错误主要出现在类加载器上

首先,出现这个问题,因为类的全名和路径都一样,那么根据java判断两个类是否是同一个类的2个原则:全名和类加载器,所以这个问题肯定是由类加载器不同引起的。由于使用 devtools 开启了热部署后,RestartClassLoader 目的是用于开发的时候,修改了类之后项目的快速重启和重载,导致了两个类无法转换

Stackoverflow 上的求助 也没有提供有效的解决办法

解决方式一

不开启 devtools ,也就是热部署,注释代码


<!-- 
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-devtools</artifactId>
  <optional>true</optional>
</dependency>
-->


这种办法最直接最快,但是体验最不好,全网搜出来也差不多是这种,注释后,无法热部署,开发很头疼,这种方式坚决不推荐!

解决方式二

SpringBoot Devtools 官方文档有说明,自定义重启类加载器

创建一个 META-INF/spring-devtools.properties 文件:

redis 存的对象为什么自动转json了 redis类型转换异常_mybatis

该 spring-devtools.properties 文件可以包含以 restart.exclude 和开头的属性 restart.include 。该include元素是应该被拉高到“重启”的类加载器的项目,以及 exclude 要素是应该向下推入“基地”类加载器的项目。该属性的值是一个应用于类路径的正则表达式模式,如以下示例所示:


restart.exclude.companycommonlibs=/mycorp-common-[\\w\\d-\.]+\.jar restart.include.projectcommon=/mycorp-myproj-[\\w\\d-\.]+\.jar


第一时间想到的是加入以下代码:


restart.include.cache=/spring-data-redis-.*.jar


配置 redis 相关类的重启,但是报错了,项目都启动不了。

后来,转头一想,应该是从 redis 取出缓存使用 GenericFastJsonRedisSerializer 做反序列化的时候转换类型报错,所以应该将 fastjson include 重启中,如下:


restart.include.json=/com.alibaba.fastjson.*.jar


重启项目后就可以解决 SpringBoot2 开启 spring-boot-devtools 插件热部署 使用 springboot-data-redis 在反序列化的时候报错 java.lang.ClassCastException: DTOObject cannot be cast to DTOObject 的问题了

此处我使用的是:GenericFastJsonRedisSerializer 做序列化和反序列化,其它类视情况更改