XXX cannot be cast to java.util.Map

博客讲述使用Hibernate查询Oracle数据库,返回自定义pojo类型时遇到类型转换异常,原因是Oracle返回字段默认为大写。通过自定义ResultTransformer类,即IgnoreCaseResultTransformer类解决该问题,并给出参考代码和使用方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

数据库为oracle,在使用Hibernate查询时想要返回自定义的pojo类型,遇到类型转换的异常。如下面的命名查询希望返回数据行为typeClass类型

query.setResultTransformer(Transformers.aliasToBean(McnInfoVo.class));

return query.getResultList();

运行后报错,错误信息如下:

McnInfoVo类如下:

@Data
public class McnInfoVo implements Serializable {
    private static final long serialVersionUID = -621560689577743929L;

    /**
     * 节目ID
     */
    private String programId;
    
    /**
     * 投放渠道
     */
    private String disposeChannel;
    
    /**
     * 应用渠道
     */
    private String applyChannel;
    
    /**
     * 开始时间
     */
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
    @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime startTime;
    
    /**
     * 结束时间
     */
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
    @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime endTime;

    /**
     * 操作人账号
     */
    private String operator;

    /**
     * 操作人昵称
     */
    private String operatorName;
    
    /**
     * 阶段状态
     */
    private String nodeStatus;
    
}
 

原因是使用oracle时返回的字段默认为大写,导致转换失败。Hibernate可以通过自定义ResultTransformer类来解决这个问题,给出参考代码:

import java.lang.reflect.Field;
import java.util.List;
import org.apache.commons.beanutils.BeanUtilsBean;
import org.hibernate.HibernateException;
import org.hibernate.transform.ResultTransformer;
 
/**
 * 
 * 修正hibernate返回自定义pojo类型时找不到属性的BUG
 * 主要发生在使用oracle时,查询返回的字段默认是大写的(除非SQL中指定了别名),这导致返回自定义pojo类型时会报找不到属性的错误,该类用于修正此BUG。
 * 使用该类时SQL返回的字段名大小写或者带"_"都会被忽略,如数据库字段为 USER_NAME,自定义pojo的属性名为username就可以使用
 * 
 *
 */
public class IgnoreCaseResultTransformer implements ResultTransformer {
 
    private static final long serialVersionUID = -3779317531110592988L;
 
    private final Class<?> resultClass;
    private Field[] fields;
    private BeanUtilsBean beanUtilsBean;
 
    public IgnoreCaseResultTransformer(final Class<?> resultClass) {
        this.resultClass = resultClass;
        this.fields = this.resultClass.getDeclaredFields();
        beanUtilsBean=BeanUtilsBean.getInstance();
    }
 
    /**
     * aliases为每条记录的数据库字段名,ORACLE字段名默认为大写
     * tupe为与aliases对应的字段的值
     */
    public Object transformTuple(final Object[] tuple, final String[] aliases) {
        Object result;
        try {
            result = this.resultClass.newInstance();
            for (int i = 0; i < aliases.length; i++) {
                for (Field field : this.fields) {
                    String fieldName = field.getName();
                    //数据库字段带下划线的时候也能保证使用,如数据库字段为 USER_NAME,自定义pojo的属性名为username就可以使用
                    if (fieldName.equalsIgnoreCase(aliases[i].replaceAll("_", ""))) {
                        beanUtilsBean.setProperty(result, fieldName, tuple[i]);
                        break;
                    }
                }
            }
        } catch (Exception e) {
            throw new HibernateException("Could not instantiate resultclass: " + this.resultClass.getName(), e);
        }
        return result;
    }
 
    @SuppressWarnings("rawtypes")
    public List transformList(final List collection) {
        return collection;
    }
}

 

使用方法:

query.setResultTransformer(new IgnoreCaseResultTransformer(McnInfoVo.class));
return query.getResultList();

完美解决!

<think>好的,我现在需要解决用户的问题:Java中出现java.lang.String cannot be cast to java.util.Map的类型转换异常。用户提供了一些引用内容,我需要仔细分析这些引用,然后找到解决方案。 首先,用户提供的引用[1]中提到,使用ObjectMapper的readValue方法将字符串转换为List<Object>。这说明正确的做法应该是使用JSON反序列化,而不是直接强制转换。而直接强制转换(String) map.get("rows")会导致类型不匹配,因为字符串无法直接转为Map或List。引用[2]和[3]指出,当尝试将Integer或String强制转换时出现异常,说明类型不匹配的问题普遍存在,不能依赖强制转换。引用[4]提到SparkR读取CSV时的类转换异常,但可能和当前问题关联不大,但进一步确认是否涉及数据解析的问题。 用户的问题核心在于如何将字符串转换为Map类型。直接使用强制转换如(Map<String, Object>) someString显然是不行的,因为String和Map是不同的类型,无法通过强制转换实现。正确的做法应该是使用JSON解析库,比如Jackson的ObjectMapper,将JSON格式的字符串反序列化为Map对象。例如,引用[1]中的示例展示了如何将字符串转换为List<Object>,同样的方法可以应用于Map。 接下来,我需要整理出解决步骤。首先,确认字符串是否为有效的JSON格式,这是反序列化的前提。然后,引入正确的依赖库,如Jackson。之后,使用ObjectMapper的readValue方法,指定目标类型为Map,可能需要使用TypeReference来保留泛型信息。同时,需要处理可能的异常,如JsonProcessingException。另外,检查代码中是否有其他地方错误地进行了强制转换,避免类似问题。 可能的相关问题包括如何选择JSON库、处理复杂嵌套结构、异常处理的最佳实践等。需要确保用户了解不能直接强制转换,而是必须通过解析的方式。同时,提醒用户注意JSON格式的正确性,避免解析失败。总结解决方案时,分步骤说明,并给出代码示例,这样用户能更清楚如何操作。</think>### 解决方案:Java中`String`无法转换为`Map`的类型转换异常 #### 问题原因分析 出现`java.lang.String cannot be cast to java.util.Map`异常的直接原因是尝试将字符串对象强制转换为`Map`类型。例如: ```java Map<String, Object> map = (Map<String, Object>) jsonString; // 错误! ``` 字符串和`Map`是两种完全不同的数据类型,直接强制转换会因类型不兼容而抛出`ClassCastException`[^1][^3]。 --- #### 解决方案步骤 ##### 1. **使用JSON解析库反序列化** 若字符串是JSON格式,需通过JSON解析库(如**Jackson**或**Gson**)将其反序列化为`Map`对象。 **示例代码(Jackson库)**: ```java import com.fasterxml.jackson.core.type.TypeReference; import com.fasterxml.jackson.databind.ObjectMapper; public class JsonUtils { private static final ObjectMapper mapper = new ObjectMapper(); public static Map<String, Object> parseJsonToMap(String json) { try { return mapper.readValue(json, new TypeReference<Map<String, Object>>() {}); } catch (JsonProcessingException e) { throw new RuntimeException("JSON解析失败", e); } } } ``` 调用方式: ```java String jsonString = "{\"key\": \"value\"}"; Map<String, Object> map = JsonUtils.parseJsonToMap(jsonString); ``` ##### 2. **检查数据来源的泛型定义** 若数据从其他接口获取(如`Map.get("key")`),需确认实际存储的类型是否为`String`,而非`Map`。例如: ```java // 错误示例:假设map.get("rows")返回的是String类型,但尝试强制转换为Map Map<String, Object> map = (Map<String, Object>) map.get("rows"); // 正确做法:先转换为String,再解析 String json = (String) map.get("rows"); Map<String, Object> parsedMap = JsonUtils.parseJsonToMap(json); ``` ##### 3. **避免直接强制转换** 所有涉及泛型容器(如`Map`、`List`)的类型转换,必须通过反序列化完成[^2]。 --- #### 相关工具与注意事项 1. **JSON库选择**: - **Jackson**:高性能,适合复杂场景(需添加依赖`com.fasterxml.jackson.core:jackson-databind`) - **Gson**:简单易用(依赖`com.google.code.gson:gson`) 2. **嵌套结构处理**: ```java // 解析嵌套JSON(如Map中嵌套List) TypeReference<Map<String, List<Object>>> type = new TypeReference<>() {}; Map<String, List<Object>> result = mapper.readValue(json, type); ``` 3. **异常处理**: 使用`try-catch`捕获`JsonProcessingException`,避免程序因格式错误的JSON崩溃。 ---
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值