2012-05-30 36 views
0

我有一個自定義覆蓋和一堆標記的地圖。因爲我在縮放改變發生時做了標記分組,所以我必須清除覆蓋圖,重新分組項目以進行新的縮放,然後再次放置在地圖上。我在後臺線程中做這件事,以使事情順利。所以你點擊放大,標記消失,繁忙的指標顯示在角落,新的標記出現在地圖上。所有的都是平滑的,因爲它應該但是當你在行中放大兩倍或放大和立即縮小時,我會得到一致的修改異常。我已經讀了很多關於這個,但沒有解決方案對我來說還可以。例如,移動地圖moification到UI上的onPostExecute可以防止這種情況發生,但會凍結地圖,繪製時間標記並且失去平滑度。我試圖在逐項疊加同步列表上創建標記列表,但沒有幫助。我怎樣才能防止這一點?Android maps item itemoverlay ConcurrentModificationException

這就是我所謂的縮放變化:

private void drawItems() { 
    final MyMapView mw = this.mapView; 

    //remove existing first 
    //mw.getOverlays().remove(stuffOverlay); 
    stuffOverlay.clearAll(); 
    //mw.invalidate(); 

    new AsyncTask<String, Integer, String>() { 
    @Override 
    protected void onPreExecute() { 
     super.onPreExecute(); 
     ((ProgressBar)findViewById(R.id.mapProgressBar)).setVisibility(View.VISIBLE); 
    } 

    @Override 
    protected String doInBackground(String... params) { 
     //get items on map 
     HashMap<String, JSONArray> groupedItems = getItemsForZoom(mw.getZoomLevel()); 

     if (groupedItems==null) 
     return "done"; 

     //create a copy of it sometimes it was throwing concurentmodificationexception 
     HashMap<String, JSONArray> groupedItemsCopy = new HashMap<String, JSONArray> (groupedItems); 

     Iterator it = groupedItemsCopy.entrySet().iterator(); 
     while (it.hasNext()) { //loop over markers 
     try { 
      //get item 
      HashMap.Entry<String, Object> pair = (HashMap.Entry)it.next(); 
      JSONArray itemsInMarker = (JSONArray) pair.getValue(); 
      JSONObject itemData = (JSONObject) itemsInMarker.get(0); 

      //get truncated coordinates 
      String coordsKey = pair.getKey(); 
      String[] coordsSplitted = coordsKey.split("_"); 

      //add first marker to map 
      GeoPoint gp = new GeoPoint((int)(Double.parseDouble(coordsSplitted[0])*1E6), (int)(Double.parseDouble(coordsSplitted[1])*1E6)); 
      OverlayItem markerOverlay = new OverlayItem(gp, "marker", "snipper text"); 
      Drawable markerPic = ServerCall.drawableFromUrl(((JSONObject)((JSONObject)itemData.get("picture")).get("1")).get("picture_full")+"_0.png"); 
      //markerPic.setBounds(0, 0, markerPic.getIntrinsicWidth(), markerPic.getIntrinsicHeight()); 
      markerPic.setBounds(-25, -25, 50, 50); 
      markerOverlay.setMarker(markerPic); 
      stuffOverlay.addOverlay(markerOverlay, itemData); 

      if (stuffOverlay.size() > 0) { 
      mapView.getOverlays().add(stuffOverlay); 
      } 
     } catch (JSONException e) { 
      e.printStackTrace(); 
     } catch (IOException e) { 
      e.printStackTrace(); 
     } 
     } 
     return "done"; 
    } 

    @Override 
    protected void onPostExecute(String result) { 
     super.onPostExecute(result); 
     ((ProgressBar)findViewById(R.id.mapProgressBar)).setVisibility(View.GONE); 
     mapView.invalidate(); 
    } 
    }.execute(); 
} 

這是我的逐項疊加:

private class StuffOverlay extends ItemizedOverlay { 
    private List<OverlayItem> mOverlays = Collections.synchronizedList(new ArrayList<OverlayItem>()); 
    private ArrayList<JSONObject> mStuff = new ArrayList<JSONObject>(); 
    public int currentItem; 
    Map mMap; 

    public StuffOverlay(Drawable defaultMarker, Map map) { 
    super(boundCenterBottom(defaultMarker)); 
    mMap = map; 
    populate(); //keep here 
    } 

    public void addOverlay(OverlayItem overlay, JSONObject stuffData) { 
    synchronized (mOverlays) { 
     mOverlays.add(overlay); 
    } 

    mStuff.add(stuffData); 
    setLastFocusedIndex(-1); //keep here 
    populate(); 
    } 

    public void clearAll() { 
    synchronized (mOverlays) { 
     mOverlays.clear(); 
    } 

    setLastFocusedIndex(-1); //keep here 
    populate(); 
    } 

    @Override 
    protected OverlayItem createItem(int i) { 
    synchronized (mOverlays) { 
     return mOverlays.get(i); 
    } 
    } 

    @Override 
    public int size() { 
    synchronized (mOverlays) { 
     return mOverlays.size(); 
    } 
    } 

    public void draw(Canvas canvas, MapView mapView, boolean shadow) { 
    if (!shadow) { 
     super.draw(canvas, mapView, false); 
    } 
    } 
} 

而這就是我得到O(

05-30 20:45:35.485: W/dalvikvm(1139): threadid=3: thread exiting with uncaught exception (group=0x4001b188) 
05-30 20:45:35.485: E/AndroidRuntime(1139): Uncaught handler: thread main exiting due to uncaught exception 
05-30 20:45:35.495: E/AndroidRuntime(1139): java.util.ConcurrentModificationException 
05-30 20:45:35.495: E/AndroidRuntime(1139):  at java.util.AbstractList$SimpleListIterator.next(AbstractList.java:64) 
05-30 20:45:35.495: E/AndroidRuntime(1139):  at com.google.android.maps.OverlayBundle.draw(OverlayBundle.java:44) 
05-30 20:45:35.495: E/AndroidRuntime(1139):  at com.google.android.maps.MapView.onDraw(MapView.java:476) 
05-30 20:45:35.495: E/AndroidRuntime(1139):  at android.view.View.draw(View.java:6535) 
05-30 20:45:35.495: E/AndroidRuntime(1139):  at android.view.ViewGroup.drawChild(ViewGroup.java:1531) 
05-30 20:45:35.495: E/AndroidRuntime(1139):  at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1258) 

回答

1

我認爲這個問題可能與這裏這些行做:

 if (stuffOverlay.size() > 0) { 
     mapView.getOverlays().add(stuffOverlay); 
     } 

你只需要對象stuffOverlay添加到地圖疊加陣列一次,你甚至可以做到這stuffOverlay中有任何項目之前。我會將mapView.getOverlays().add(stuffOverlay);移動到stuffOverlay實例化的地步。

+0

偉大的地方,你是絕對正確的,它確實解決了我的問題。謝謝。 – Stan

相關問題