ConcurrentHashMap.keySet的兼容问题

ConcurrentHashMap.keySet的兼容问题
0

这两天遇到一个问题,我用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
}

通过这个问题,我们可以看到,在定义变量的时候用接口来定义而不是用实现类来定义,在实现类的具体实现变化的时候能避免产生一些兼容性的问题