您的当前位置:首页正文

java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String

2024-11-26 来源:个人技术集锦

java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String

问题背景

  • 接口中参数为 Map<String,Object>取出某 KEY 值时,强制将其转为 String ,出现上述异常

问题分析

  • Map<String,Object> 作为参数的原因:可以传递多个参数;若使用 PO 类作为参数,不同项目中引用时需升级版本;
  • 出现问题的原因:对外提供接口,没有在接口上明确注释,某个key的value是什么类型,只有当时写此接口的人才会避免此问题,其他调用者则不清楚,在调用接口后对结果进行了强制转换
  • 代码级别分析:问题的根本是泛型与Object的区别
// 示例:模拟将Integer存入Map,取出后强制转为String
    public static void main(String[] args) {
        Map<String,Object> params = new HashMap<String,Object>();
        // value为 int 类型
        params.put("1", 1);
        // 取出 value 转为 String 
        // 运行时此处出现
        // java.lang.Integer cannot be cast to java.lang.String
        String result = (String)params.get("1");
        System.out.println(result);

    }

问题:

  • map.get(“”) 返回类型为 Object ,为何将 Object 转为 String 时出现异常?
// java -version 1.7.0_79 源码
// 1. java.util.Map 的数据结构
// HashMap 为解决hash冲突使用链地址法:
// 整体是一个数组结构,数组中每个位置存放一个链表,Entry 为链表中的一个节点
transient Entry<K,V>[] table = (Entry<K,V>[]) EMPTY_TABLE;
// 从 Entry<K,V> 得出,其中 K,V 都是泛型

// 2. java.util.HashMap 中 put方法
public V put(K key, V value) {
}
// 方法中的参数类型均为泛型,与 Entry 对应

// 3.泛型的优势
// 编译期可确认数据类型

// 4.泛型与Object 的区别
// 使用泛型取值,无需类型转换; Object 需要转换
// 使用泛型作为参数,在编译期可检测数据类型不一致的问题
// 如:
Map<String,Object> params = new HashMap<String,Object>();
params.put("1",1);
params.put(2,2) ; // 编译期出现异常提示;K k ,指定 K 泛型的数据类型为 String 

// 5.JVM 在编译器根据执行的泛型的数据类型擦除泛型
// JVM 中无泛型的概念
// JVM 编译时根据指定的数据类型,如上例中,VALUE 为 int 类型
// JVM 编译代码后,VALUE  变为 int 类型

// 6.分析上述例子中的结果
        Object object = params.get("1");
        if(object instanceof Integer){
            System.out.println("true"); // 控制台打印 true 
            // 由输出结果可知:虽然map中取出的结果是 Object ,但其实质是 Integer ,故在将其强制转为 String 时会出现强制类型转换的异常
        }
// 示例:解决方法
        String result1 = String.valueOf(object);
        // String.valueOf 实质调用 object.toString() 方法
        System.out.println(result1);

问题总结

泛型

泛型常用表示及含义

  • E – Element (在集合中使用,因为集合中存放的是元素)
  • T – Type(Java 类)
  • K – Key(键)
  • V – Value(值)
  • N – Number(数值类型)
  • ? – 表示不确定的java类型(无限制通配符类型)
  • S、U、V – 2nd、3rd、4th types

参考资料

显示全文