java.lang.ClassCastException:DTOObject 无法转换为 DTOObject

Posted

技术标签:

【中文标题】java.lang.ClassCastException:DTOObject 无法转换为 DTOObject【英文标题】:java.lang.ClassCastException: DTOObject cannot be cast to DTOObject 【发布时间】:2016-10-24 22:25:54 【问题描述】:

我在 Spring Boot 1.4.0M3 上运行的应用程序中遇到了一个奇怪的问题,该应用程序使用 Spring 缓存实现,其中提供者是 Redis,我收到 classCastException 无法转换相同的对象

我使用 Mongodb 作为数据库,我有用户对象,其中包含延迟加载的角色对象列表,角色内部包含权限对象,如下所示

@Document
@Data
public class User implements Serializable
private String passwordResetToken;

private boolean enabled = false;

@DBRef(lazy= true)
private List<Role> roleList;

我的角色 DTO 如下

@Data
@Document
public class Role implements Serializable
   private String roleName;
    private String description;
    @DBRef(lazy= true)
    private List<Permission> permissions;

现在在我的 spring MVC 中,在加载所有角色时,我正在调用所有权限,因为这是重复操作,我想缓存结果并使用 redis,并在加载角色值时收到异常。

raised java.lang.ClassCastException: com.learning.securedapp.domain.Permission cannot be cast to com.learning.securedapp.domain.Permission

帮助我克服这个错误。

我将source code 附加到我的项目中,我在RoleController.java 的第91 行收到错误

要在本地环境中复制,登录到应用程序并单击权限菜单,然后单击角色菜单,现在在角色菜单中单击任何编辑图标。您将收到上述错误。

【问题讨论】:

如果一个类被不同的类加载器加载了两次,jvm 认为它们是不相等的。我想这就是发生在你身上的事情。 看起来你正在使用一个war文件,所以问题可能在于我们的应用程序容器加载了同一个类的多个版本。 我从 ide 运行,我只看到一个版本的权限类。 【参考方案1】:

当您将 DevTools 与缓存一起使用时,您需要注意 this limitation。

当对象被序列化到缓存中时,应用类加载器是C1。然后在您更改一些代码/配置后,devtools 会自动重新启动上下文并创建一个新的类加载器 (C2)。当您点击该缓存方法时,缓存抽象会在缓存中找到一个条目,并将其从存储中反序列化。如果缓存库不考虑上下文类加载器,则该对象将附加错误的类加载器(这解释了奇怪的异常A cannot be cast to A)。

TL;DR 如果缓存库不使用上下文类加载器,则不要使用 devtools 序列化类。或者把你的缓存库in the application classloader:

restart.include.yourcache=/my-cache-lib-[\\w-]+\.jar

【讨论】:

感谢@nicoll,能否提供一个示例,说明如何在应用程序类加载器中排除缓存库 第二个链接有一个例子。您实际上应该在应用程序类加载器中包含缓存库。我会更新问题。 我已经按照您提供的链接并在 META-INF/spring-devtools.properties 下添加了 restart.exclude.cache=jedis*.jar 属性,但我仍然面临同样的问题。我添加了 jedis,因为我使用 redis 作为我的缓存提供程序 我认为连载的不是绝地武士。 更改为 restart.include.cache 也不起作用【参考方案2】:

这对我有用,DevTools 和 Redis 都在工作。我们需要在创建 JdkSerializationRedisSerializer 时传递 classLoader,它应该可以工作

 JdkSerializationRedisSerializer redisSerializer = new JdkSerializationRedisSerializer(getClass().getClassLoader());

所以我的 RedisCacheConfig 是:

@Configuration
@EnableCaching
public class RedisCacheConfig extends CachingConfigurerSupport implements CachingConfigurer 


............................
............................


    @Bean
public RedisCacheManager redisCacheManager(LettuceConnectionFactory lettuceConnectionFactory) 
    JdkSerializationRedisSerializer redisSerializer = new JdkSerializationRedisSerializer(getClass().getClassLoader());

    RedisCacheConfiguration redisCacheConfiguration = RedisCacheConfiguration.defaultCacheConfig()
            .disableCachingNullValues()
            .entryTtl(Duration.ofHours(redisDataTTL))
            .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(redisSerializer));

    redisCacheConfiguration.usePrefix();

    RedisCacheManager redisCacheManager = RedisCacheManager.RedisCacheManagerBuilder.fromConnectionFactory(lettuceConnectionFactory)
            .cacheDefaults(redisCacheConfiguration)
            .build();

    redisCacheManager.setTransactionAware(true);
    return redisCacheManager;


............................
............................



检查这个 Spring Boot 问题:https://github.com/spring-projects/spring-boot/issues/9444

【讨论】:

【参考方案3】:

我实际上尝试了建议的解决方案(及其许多变体),但没有成功。例如,这并没有阻止问题的发生:

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

我更新了上述内容以标注我正在使用的特定版本,但它仍然无法正常工作。

我最终所做的工作是将 spring-boot-devtools 从我的项目中排除。我正在使用 Maven,所以注释是这样的:

    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-devtools</artifactId>
        <version>[1.5.9,)</version>
        <scope>provided</scope>
    </dependency>

这将阻止任何等于或大于 1.5.9 的版本加载。在我包含上述内容后,一切都按预期工作。我知道这对所有人来说都不是一个理想的解决方案,但我很少使用 devtools 的重启功能,所以这对我来说实际上是一个好方法。

【讨论】:

【参考方案4】:

我使用的是 Spring Boot 2.0.5,我最终从 pom.xml 中完全删除了 devtools。感谢@Always Learning 的上述回答。 尽管我很讨厌这样做,但我现在找不到其他方法!

  <!-- 
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <scope>runtime</scope>
        </dependency>
     -->    

【讨论】:

以上是关于java.lang.ClassCastException:DTOObject 无法转换为 DTOObject的主要内容,如果未能解决你的问题,请参考以下文章