我创建了osmdroid地图视图,我在其上显示自定义WMS,这是有效的。接下来,我需要在2点之间的地图路线上绘制dinamically,我从GeoJSON格式的自定义服务获得。 GeoJSON看起来像这样:http://pastebin.com/GJWYNkAq
通过OkHttp客户端调用服务:
Request request = new Request.Builder()
.url("http://xxx.xxx.xxx.x:7915/GeoService.svc/GetRoute?" + "source=" + encodedSourceAddress + "&target=" + encodedTargetAddress)
.build();
client.newCall(request).enqueue(new Callback() {
@Override
public void onFailure(Call call, IOException e) {
e.printStackTrace();
}
@Override
public void onResponse(Call call, Response response) throws IOException {
if (!response.isSuccessful())
throw new IOException("Unexpected code " + response);
String result = response.body().string();
Log.d("RESULT", result);
viewRoute(result); //json is received ok, I debugged it
}
});
我在这个方法中使用了这个Osmbonuspack库:
public void viewRoute(String geoJson) {
KmlDocument kmlDocument = new KmlDocument();
kmlDocument.parseGeoJSON(geoJson); //application is crashed here
FolderOverlay myOverLay = (FolderOverlay) kmlDocument.mKmlRoot.buildOverlay(map, null, null, kmlDocument);
map.getOverlays().add(myOverLay);
map.invalidate();
}
当我运行app时,启动后崩溃时出现此错误:
03-09 13:01:51.521 3968-3992 / bachelor.vsb.martin.osmdroidclient E / AndroidRuntime:FATAL EXCEPTION:OkHttp Dispatcher 流程:bachelor.vsb.martin.osmdroidclient,PID:3968 java.lang.ClassCastException:无法转换com.google.gson.JsonNull 到com.google.gson.JsonObject 在com.google.gson.JsonObject.getAsJsonObject(JsonObject.java:191) 在 org.osmdroid.bonuspack.kml.KmlPlacemark。(KmlPlacemark.java:89) 在 org.osmdroid.bonuspack.kml.KmlFeature.parseGeoJSON(KmlFeature.java:237) 在org.osmdroid.bonuspack.kml.KmlFolder。(KmlFolder.java:62) 在 org.osmdroid.bonuspack.kml.KmlFeature.parseGeoJSON(KmlFeature.java:235) 在 org.osmdroid.bonuspack.kml.KmlDocument.parseGeoJSON(KmlDocument.java:1097) 在 org.osmdroid.bonuspack.kml.KmlDocument.parseGeoJSON(KmlDocument.java:1112) 在 bachelor.vsb.martin.osmdroidclient.MainActivity.viewRoute(MainActivity.java:137) 在 bachelor.vsb.martin.osmdroidclient.MainActivity $ 1.onResponse(MainActivity.java:127) 在okhttp3.RealCall $ AsyncCall.execute(RealCall.java:135) 在okhttp3.internal.NamedRunnable.run(NamedRunnable.java:32) 在 java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1133) 在 java.util.concurrent.ThreadPoolExecutor中的$ Worker.run(ThreadPoolExecutor.java:607) 在java.lang.Thread.run(Thread.java:761)
我正在使用osmdroid 5.6.4和osmbonuspack 6.2
答案 0 :(得分:1)
它看起来像是该库中的一个错误。 master分支的current实现如下所示:
89: JsonObject geometry = json.getAsJsonObject("geometry");
90: if (geometry != null) {
是的,getAsJsonObject
可以针对不存在的密钥返回null
,但是如果给定的JSON路径,Gson使用JsonNull
作为null
值标记(并且它是有意义的)值已知为null
。 JSON中唯一具有null
的路径是$.features[6].geometry
,所以我假设这个路径是唯一的原因。你可以在这里做的是在他们的GitHub问题台提出一个新问题,添加一个适当的JsonNull
检查,以免在类型转换中失败。模拟案例:
final JsonObject geometry = (JsonObject) geoJsonObject.getAsJsonObject()
.get("features")
.getAsJsonArray()
.get(6)
.getAsJsonObject()
.get("geometry"); // actually JsonNull for $.features[6].geometry
到目前为止,你可以做的是:
geometry
JsonPath
元素,因此可能只是替换该路径值赢得了&#39 ;工作)。feature
对象(这可能更合适,但会导致一些数据丢失)。第一个选项的示例:
final JsonObject geoJsonObject = gson.fromJson(geoJson, JsonObject.class);
fixGeoJsonObject(geoJsonObject);
kmlDocument.parseGeoJSON(geoJsonObject); // passing the "fixed" object
其中fixGeoJsonObject
如下:
private static void fixGeoJsonObject(final JsonObject geoJsonObject) {
final JsonArray features = geoJsonObject
.get("features")
.getAsJsonArray();
final int length = features.size();
for ( int i = 0; i < length; i++ ) {
final JsonObject feature = features.get(i)
.getAsJsonObject();
final JsonElement geometry = feature.get("geometry");
if ( geometry.isJsonNull() ) {
feature.remove("geometry"); // losing some data...
}
}
}
或第二个选项:
private static void fixGeoJsonObject(final JsonObject geoJsonObject) {
final JsonArray features = geoJsonObject
.get("features")
.getAsJsonArray();
for ( int i = 0; i < features.size(); i++ ) {
final JsonObject feature = features.get(i)
.getAsJsonObject();
final JsonElement geometry = feature.get("geometry");
if ( geometry.isJsonNull() ) {
features.remove(i); // losing even more data...
}
}
}
这应该可以解决您获得的ClassCastException
,但我不知道它会如何影响您正在使用的库。