在安卓谷歌地图中展示大型GeoJSON文件

10
我试图将一个geojson层显示在Google地图上。代码如下。geojson文件存储在我的raw文件夹中。该文件有286个特征(约15MB)。因此,读取并显示它需要消耗更多的时间。最初,我遇到了内存不足的错误,通过在清单文件中设置大堆为true来解决了这个问题。如何快速加载此文件(当前需要一分钟或更长时间)?我想知道是否有其他有效的方法来做到这一点。之后,我还要执行其他任务,如获取功能并显示一些属性。
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    SupportMapFragment mapFragment = (SupportMapFragment) getSupportFragmentManager().findFragmentById(R.id.map);
    mapFragment.getMapAsync(this);
}

@Override
public void onMapReady(GoogleMap googleMap) {
    mMap = googleMap;
    mMap.moveCamera(CameraUpdateFactory.newLatLngZoom(myanmar, 5.25f));
    new MyAsyncTask().execute();
}

public class MyAsyncTask extends AsyncTask <Void, Void, Void> {

    ProgressDialog pd;
    GeoJsonLayer layer;

    @Override
    protected void onPreExecute() {
        super.onPreExecute();
        pd = new ProgressDialog(MapsActivity.this);
        pd.setMessage("Loading Data");
        pd.show();
    }

    @Override
    protected Void doInBackground(Void... voids) {
        try {
            layer = new GeoJsonLayer(mMap, R.raw.myanmar, getApplicationContext());
        } catch (Exception e) {
            Log.d("Error is : ", e.toString());
        }
        return null;
    }

    @Override
    protected void onPostExecute(Void aVoid) {
        super.onPostExecute(aVoid);
        pd.cancel();
        layer.addLayerToMap();
     }
}

我的GeoJSON文件的一个非常小的部分如下:

{
 "type": "FeatureCollection",
 "crs": { "type": "name", "properties": { "name": "urn:ogc:def:crs:OGC:1.3:CRS84" } },
 "features": [
{ "type": "Feature", "properties": { "ID_3": 193, "NAME_3": "Chaung-U" }, "geometry": { "type": "MultiPolygon", "coordinates": [ [ [ [ 95.348327636718807, 21.878610610961914 ], [ 95.297210693359432, 21.860000610351676 ], [ 95.286926269531307, 21.853612899780387 ], [ 95.276092529296989, 21.850000381469727 ], [ 95.265823364257926, 21.84832954406744 ], [ 95.257217407226676, 21.846940994262695 ] ] ] ] } },
{ "type": "Feature", "properties": { "ID_3": 199, "NAME_3": "Myaung" }, "geometry": { "type": "MultiPolygon", "coordinates": [ [ [ [ 95.376281738281193, 21.708541870117301 ], [ 95.375259399414119, 21.703050613403434 ], [ 95.370529174804801, 21.681390762329102 ], [ 95.367752075195313, 21.664720535278434 ], [ 95.366699218750114, 21.658369064331055 ], [ 95.362762451171875, 21.649999618530273 ] ] ] ] } }
]}

我尝试将文件分成多个小文件单元,并并行地运行单独的异步任务,每个任务处理一个单独的文件:

for (int i = 1; i < 3; i++) {
        new MyAsyncTask().executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR,i);
}

异步任务(AsyncTask):

public class MyAsyncTask extends AsyncTask <Integer, Void, GeoJsonLayer> {

    @Override
    protected void onPreExecute() {
        super.onPreExecute();
    }

    @Override
    protected GeoJsonLayer doInBackground(Integer... integers) {
        int i = integers[0];
        try {
            GeoJsonLayer layer = null;
            switch (i) {
                case 1:
                    layer = new GeoJsonLayer(mMap, R.raw.small_1,
                            getApplicationContext());
                    break;
                case 2:
                    layer = new GeoJsonLayer(mMap, R.raw.small_2,
                            getApplicationContext());
                    break;
                //other cases
            }
            for (GeoJsonFeature feature : layer.getFeatures()) {
                geoJsonFeatures.put(Integer.parseInt(feature.getProperty("ID_3")), feature);
            }
            return layer;
        } catch (Exception e) {
            return null;
        }
    }

    @Override
    protected void onPostExecute(GeoJsonLayer layer) {
        super.onPostExecute(layer);
        layer.addLayerToMap();
    }
}

这个建议可行吗?这么做可以比之前缩短时间,但考虑到用户体验因素,速度仍然不够快。


1
你能否研究一下谷歌的GSON库,它具有流式处理功能,可以在不过多影响内存的情况下加载大量JSON数据。https://github.com/google/gson。这里可以找到一个快速示例:https://www.mkyong.com/java/gson-streaming-to-read-and-write-json/。 - Tixeon
@Sanny 我研究了使用Gson(https://sites.google.com/site/gson/gson-user-guide),但我在解析我的geojson文件时遇到了问题。我在问题中添加了我的geojson文件的示例。此外,我不确定如何在反序列化后在Android Google地图上显示它。 - Sujal
3个回答

5

最终通过减小geojson文件的大小解决了问题。使用http://mapshaper.org/这个在线工具,提供了可以减小文件大小但保持形状的选项。使用它获得了简化后的文件(近300KB),加载在几秒钟内完成。希望这能帮助到遇到相同问题的人。


我有一个非常大的GeoJSON,渲染在iOS和Android上时会崩溃。 - famfamfam
嗨。数据正在从服务器加载并需要转换为geojson格式,所以我不知道如何调整大小。您能否在此处查看格式:https://pastebin.com/VxSYqvum - famfamfam
在这种情况下,也许您的服务器可以向您发送最佳功能以在地图中显示,这样您就不必调整大小。要创建geojson,您可以尝试使用https://github.com/cocoahero/android-geojson - Sujal

1
我已经编写了如何反序列化GeoJSON代码的代码。但是我以前从未尝试过Google地图。也许你可以提出另一个问题?下面我将向您展示如何使用GSON Streaming反序列化您的大型GeoJSON文件。

GeoJSON.java

public class GeoJSON {

    @SerializedName("type")
    private String type;
    @SerializedName("features")
    private List<Features> features;

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }


    public List<Features> getFeatures() {
        return features;
    }

    public void setFeatures(List<Features> features) {
        this.features = features;
    }
}

Features.java

public class Features {

    @SerializedName("type")
    private String type;

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }
}

将JSON转换为POJO类
private void readGeoJSON() {
        try {
            InputStream is = getAssets().open("geo.json");

            JsonReader reader = new JsonReader(new InputStreamReader(is, "UTF-8"));
            Gson gson = new GsonBuilder().create();

            GeoJSON geoJSON = new GeoJSON();
            ArrayList<Features> features = new ArrayList<>();

            reader.beginObject();
            while (reader.hasNext()){
                String key = reader.nextName();
                if(key.equals("type")){
                    geoJSON.setType(reader.nextString());
                }else if(key.equals("features")){
                    // read array
                    reader.beginArray();
                    while (reader.hasNext()) {
                        Features o = gson.fromJson(reader, Features.class);
                        features.add(o);
                    }
                    reader.endArray();
                }else{
                    reader.skipValue(); //avoid some unhandle events
                }

                geoJSON.setFeatures(features);
            }

            reader.endObject();
            reader.close();


        } catch (IOException e) {
            e.printStackTrace();
        }
    }

使用这种方法,我们将所有对象逐个加载到内存中,而不是一次性加载整个文件。

假设我在资产文件夹中存储geo.json文件。以下是您如何反序列化本地的geo.json文件,我已经测试了代码,它可以正常工作。您可能需要为其他字段添加内容。希望它能解决您的内存超限问题。


我已经按照问题中的编辑所提到的另一种方式进行了尝试。我也会尝试您的反序列化代码。@Sanny - Sujal

0

@harris,您能告诉我使用异步任务的方式(在问题中进行了编辑)是否是一个好的方式吗?对于我的情况来说,什么是更有效的方式呢?也许反序列化需要额外的处理才能在地图上显示。 - Sujal
你能提供在Swift 4中使用的方法吗?谢谢。 - famfamfam

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接