分享

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

 一本正经地胡闹 2019-05-28

Map map = new HashMap();

map.put("index",1);

Map<String,String> indexmap = new HashMap<String,String>();

indexmap = map;

Object index = indexmap.get("index");一般这里会用强转:

(String)indexmap.get("index");

index其实是int类型,但二次开发人员会认为是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);

问题总结

  1. Map<String,Object>作为接口参数类型时要注明可以接收的KEY与对应VALUE的数据类型

  2. Map<String,Object> params取值时,要注意可能出现转换异常,为避免出现上述问题,可通过 instanceOf 判断;可使用toString() / String.valueOf() 将其先转为 String 再转为需要的数据类型

泛型

泛型常用表示及含义

  • E – Element (在集合中使用,因为集合中存放的是元素)

  • T – Type(Java 类)

  • K – Key(键)

  • V – Value(值)

  • N – Number(数值类型)

  • ? – 表示不确定的java类型(无限制通配符类型)

  • S、U、V – 2nd、3rd、4th types

参考资料

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多