iOS не удаляет оверлеи MKMapView - PullRequest
10 голосов
/ 16 июня 2011

Я хотел бы удалить все наложения моей карты в одной точке, и я пробовал разные способы, но это никогда не работает.

Последняя попытка, которую я сделал [self.mapView removeOverlays:self.mapView.overlays];, и она все еще не работает. Любая идея, как я могу удалить эти наложения?

Спасибо.

ОБНОВЛЕНИЕ 1

У меня ошибка: malloc: *** error for object 0x5adc0c0: pointer being freed was not allocated *** set a breakpoint in malloc_error_break to debug

Думаю, я знаю почему, но не знаю, как это исправить ... Вот код, когда мне нужно нарисовать еще одну линию на моем mapView:

// Create a c array of points. 
MKMapPoint *pointsArray = malloc(sizeof(CLLocationCoordinate2D) * 2);

// Create 2 points.
MKMapPoint startPoint = MKMapPointForCoordinate(CLLocationCoordinate2DMake(oldLatitude, oldLongitude));
MKMapPoint endPoint = MKMapPointForCoordinate(CLLocationCoordinate2DMake(newLatitude, newLongitude));

// Fill the array.
pointsArray[0] = startPoint;
pointsArray[1] = endPoint;

// Erase polyline and polyline view if not nil.
if (self.routeLine != nil) {
    [_routeLine release];
    self.routeLine = nil;
}

if (self.routeLineView != nil) {
    [_routeLineView release];
    self.routeLineView = nil;
}

// Create the polyline based on the array of points.
self.routeLine = [MKPolyline polylineWithPoints:pointsArray count:2];

// Add overlay to map.
[self.mapView addOverlay:self.routeLine];

// clear the memory allocated earlier for the points.
free(pointsArray);

// Save old coordinates.
oldLatitude = newLatitude;
oldLongitude = newLongitude;

Итак, я освобождаю объект routeLine, который является предыдущим наложением. Поэтому, когда я пытался удалить его, он падает, потому что он уже был освобожден.

Вот код делегата mapView для добавления наложенных видов:

- (MKOverlayView *)mapView:(MKMapView *)mapView viewForOverlay:(id <MKOverlay>)overlay
{
    MKOverlayView* overlayView = nil;

    if(overlay == _routeLine) {
        // If we have not yet created an overlay view for this overlay, create it now. 
        if(self.routeLineView == nil) {
            self.routeLineView = [[[MKPolylineView alloc] initWithPolyline:_routeLine] autorelease];

            self.routeLineView.fillColor = [UIColor blueColor];
            self.routeLineView.strokeColor = [UIColor blueColor];

            // Size of the trace.
            self.routeLineView.lineWidth = routeLineWidth;
        }

        overlayView = self.routeLineView;
    }

    return overlayView;
}

Если вы, ребята, знаете, как я могу решить эту проблему, удалив все наложения из моего MKMapView, это было бы здорово!

ОБНОВЛЕНИЕ 2

Я пытался не отпускать свои routeLine и routeLineView объекты, и теперь это работает. Кажется, что нет никаких утечек тоже. Итак, теперь я делаю это:

// Erase polyline and polyline view if not nil.
if (self.routeLine != nil) {
    //[_routeLine release];
    self.routeLine = nil;
}

if (self.routeLineView != nil) {
    //[_routeLineView release];
    self.routeLineView = nil;
}

Ответы [ 3 ]

11 голосов
/ 16 июня 2011

Когда вы вызываете removeOverlays:, представление карты высвобождает объекты MKOverlay и MKOverlayView.

Вы храните свои собственные ссылки на них в _routeLine и _routeLineView.

ПослеremoveOverlays: вызывается, ваши переменные будут указывать на уже освобожденную память.Когда вы воссоздаете ломаную линию, вы переусердствуете, что приводит к сбою.

Поэтому удалите вызовы release:

if (_routeLine != nil) {
    [_routeLine release];  // <-- remove this
    self.routeLine = nil;
}

if (_routeLineView != nil) {
    [_routeLineView release];  // <-- remove this
    self.routeLineView = nil;
}
7 голосов
/ 16 июня 2011

Когда вы просматриваете код в отладчике, где появляется ошибка?

Одна мысль, у вас может быть проблема с циклом сохранения / выпуска для self.routeLine и self.routeLineView.Предполагая, что они являются свойствами с атрибутом retain, когда вы делаете

self.routeLine = [MKPolyline polylineWithPoints:pointsArray count:2];  

, ваш синтезированный метод доступа сохраняет новый объект MKPolyline.Этот объект также будет иметь пару retain / autorelease из вспомогательного метода, который его создал.Когда этот метод вызывается снова, и вы вызываете код

if (self.routeLine != nil) {
    [_routeLine release];
    self.routeLine = nil;
}

, вы в конечном итоге освобождаете переменную дважды, первый с явным вызовом [_routeLine release], а второй с синтезированным средством доступа на self.routeLine = nil; вызов.Он останется в памяти, но вылетит из приложения, когда ваш пул авто-релиза опустошится.

В большинстве случаев, чтобы очистить все оверлеи на MKMapView (в этом примере с именем mapView), я 'сделать что-то вроде следующего:

for (id<MKOverlay> overlayToRemove in mapView.overlays)
{
   if ([overlayToRemove isKindOfClass:[OverlayClassToRemove class]])
   {
       [mapView removeOverlay:overlayToRemove];
   }
}
1 голос
/ 06 апреля 2017

Свифт 3, расширение

extension MKMapView
{
    func removeAllOverlay(){
        for overlay:MKOverlay in self.overlays  {
            self.remove(overlay)
        }
    }

}
...