我有一个以JSON格式存在的异构映射,我想解析它并将其转换为异构映射对象(类HeterogeneousMap
)。
要解析地图,我使用了一个对象,该对象定义了地图可以拥有的所有已知键(类HeterogeneousMapStructure
)。
MapKey<T>
接口使用方法T parseValue(JsonReader jsonReader)
来解析密钥的值。
我遇到的问题是如何将解析后的值放在类型安全的异构地图对象中:
public class HeterogeneousMap {
public <T> void put(MapKey<T> mapKey, T value) {
// Put key and value in map
}
}
public interface MapKey<T> {
T parseValue(JsonReader jsonReader) throws IOException;
}
public class HeterogeneousMapStructure {
private final List<MapKey<?>> keyList;
public HeterogeneousMap parseMap(JsonReader jsonReader) {
HeterogeneousMap heterogeneousMap = new HeterogeneousMap();
// ... find matching key
MapKey<?> matchingMapKey = ...;
/*
* Compiling error:
* The method put(TestClass.MapKey<T>, T) in the type TestClass.HeterogeneousMap
* is not applicable for the arguments (TestClass.MapKey<capture#1-of ?>, capture#2-of ?)
*/
heterogeneousMap.put(matchingMapKey, matchingMapKey.parseValue(jsonReader));
return heterogeneousMap;
}
}
有没有办法解决这个问题?
答案 0 :(得分:1)
您可以为此创建一个单独的方法:
private static <T> void parseAndPut(HeterogeneousMap map, MapKey<T> key, JsonReader in) throws IOException {
map.put(key, key.parseValue(in));
}
并在您的解析图中调用此方法:
public class HeterogeneousMapStructure {
private final List<MapKey<?>> keyList;
public HeterogeneousMap parseMap(JsonReader jsonReader) {
HeterogeneousMap heterogeneousMap = new HeterogeneousMap();
// ... find matching key
MapKey<?> matchingMapKey = ...;
/*
* Compiling error:
* The method put(TestClass.MapKey<T>, T) in the type TestClass.HeterogeneousMap
* is not applicable for the arguments (TestClass.MapKey<capture#1-of ?>, capture#2-of ?)
*/
parseAndPut(heterogeneousMap, matchingMapKey, jsonReader);
return heterogeneousMap;
}
private static <T> void parseAndPut(HeterogeneousMap map, MapKey<T> key, JsonReader in) throws IOException {
map.put(key, key.parseValue(in));
}
}
这应该可行,但我没有你所有的课程,所以我无法真正测试它们。如果他们出了问题,请告诉我,我会尽力解决。
问题在于通配符类型,或者如果您能理解,则说存在类型。 Java对这些通配符来说非常愚蠢。编译器将任意两个通配符视为可能的不同类型,即使它可以将它们推断为相同类型或它们在相同类型层次结构中。在您的情况下,两个capture#X-of ?
被认为是不相关的,因此引发了编译错误。
由于问题是由于编译器无法推断,我们明确告诉它。如您所见,我提取操作并将存在类型指定为类型变量(T
),因此编译器现在可以看到类型匹配,问题就解决了。