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> 作为参数的原因:可以传递多个参数;若使用 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);
}
问题:// 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);
问题总结Map<String,Object> 作为接口参数类型时要注明可以接收的KEY与对应VALUE的数据类型
从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
参考资料
|