ConcurrentHashMap的keySet()的兼容问题
这两天遇到一个问题,我用JDK 8编译出来的程序在JRE 6上运行报ClassNotFoundException,说找不到java.util.concurrent.ConcurrentHashMap$KeySetView这个类。
原来Java 7或以下版本中是没有java.util.concurrent.ConcurrentHashMap$KeySetView这个类的。在Java 8中,ConcurrentHashMap中才新增了KeySetView这个public的内部类,而ConcurrentHashMap的keySet方法的返回值也改成了KeySetView。这样,对于以下代码:
ConcurrentHashMap<String, Object> map = new ConcurrentHashMap<String, Object>(); for (String key : map.keySet()) { // do something }
解开for-in的语法糖,JDK 8实际上会把代码看成以下的样子:
ConcurrentHashMap<String, Object> map = new ConcurrentHashMap<String, Object>(); ConcurrentHashMap.KeySetView<String, Object> temp1 = map.keySet(); for (Iterator<String> iterator = temp1.iterator(); iterator.hasNext(); ) { String key = iterator.next(); // do something }
因此,在JRE 6上执行的时候,就会找不到java.util.concurrent.ConcurrentHashMap$KeySetView这个类。
解决方法是在定义map变量的时候,用Map接口来定义,即
Map<String, Object> map = new ConcurrentHashMap<String, Object>(); for (String key : map.keySet()) { // do something }
通过这个问题,我们可以看到,在定义变量的时候用接口来定义而不是用实现类来定义,在实现类的具体实现变化的时候能避免产生一些兼容性的问题。