如果在添加注释时缩放,MKMapView 会崩溃

Posted

技术标签:

【中文标题】如果在添加注释时缩放,MKMapView 会崩溃【英文标题】:MKMapView crashing if zooming while adding annotations 【发布时间】:2012-12-17 23:41:23 【问题描述】:

看起来我遇到了一个问题,如果我在更改地图的可见区域时添加注释,我可以可靠地导致我的MKMapView 崩溃。我已经将导致它的代码简化为一个非常简单的实现,我在这里复制它:

NSMutableArray *pointAnnotationArray = [[NSMutableArray alloc] init];
MKCoordinateRegion coordRegion = [mapViewOutlet region];
float randMax = 0.1;

for (int i = 0; i < 100; i++)
    float randomDeviation1 =  (((float) (arc4random() % ((unsigned)RAND_MAX + 1)) / RAND_MAX) * randMax) - (randMax / 2);
    float randomDeviation2 =  (((float) (arc4random() % ((unsigned)RAND_MAX + 1)) / RAND_MAX) * randMax) - (randMax / 2);

    MKPointAnnotation *point = [[MKPointAnnotation alloc] init];        
    CLLocationCoordinate2D pointLocation = CLLocationCoordinate2DMake(coordRegion.center.latitude + randomDeviation1, coordRegion.center.longitude + randomDeviation2);
    [point setCoordinate:pointLocation];    
    [pointAnnotationArray addObject: point];


[mapViewOutlet addAnnotations:[NSArray arrayWithArray:pointAnnotationArray]];

我在上面所做的是在地图上添加 100 个点,这些点随机(足够)分布在 MKMapView 的可见区域内和周围。重现这一点的最简单方法是将此代码设置为在计时器上运行(例如 5 秒后),然后抓取地图并开始放大和缩小一点,直到计时器运行。一旦你感觉到它,你每次都可能会崩溃。在addAnnotations 调用之前禁用mapViewOutlet 上的用户交互似乎没有帮助。 (也许它无法在用户处于中间手势时禁用用户交互,我想这是可以理解的。)

我的另一个问题是我没有太多的运气来追踪崩溃——我没有很多经验导致我自己的代码中没有直接发生的崩溃,所以我可能会错过一些明显的跟踪它的方法,但我目前在main.mreturn UIApplicationMain 行上遇到了一个无用的断点。我最好的猜测是,MKMapView 仅绘制可见注释的方式引起了一些问题——缩放导致可见区域发生变化,同时添加注释并且地图正在尝试确定要绘制的点。

还有其他人看过吗?关于如何避免它的任何建议,而无需在我添加注释的任一端锁定我的地图与用户交互一段相当长的时间?

编辑:忘记包含调用堆栈:

#0  0x3752a944 in objc_exception_throw ()
#1  0x3869dec0 in __NSFastEnumerationMutationHandler ()
#2  0x31929d46 in -[MKAnnotationContainerView _updateAnnotationViewPerspective] ()
#3  0x3192981a in -[MKMapView _updateScrollContainerView:] ()
#4  0x3192ff66 in -[MKMapView mapViewDidDraw:] ()
#5  0x391560b4 in -[VKMapCanvas didDrawView] ()
#6  0x3914d4a8 in -[VKScreenCanvas onTimerFired:] ()
#7  0x3914b4a8 in -[VKMapCanvas onTimerFired:] ()
#8  0x3914a346 in -[VKMainLoop displayTimerFired:] ()
#9  0x35ebe780 in CA::Display::DisplayLink::dispatch(unsigned long long, unsigned long long) ()
#10 0x35ebe6d8 in CA::Display::IOMFBDisplayLink::callback(__IOMobileFramebuffer*, unsigned long long, unsigned long long, unsigned long long, void*) ()
#11 0x34815fd6 in IOMobileFramebufferVsyncNotifyFunc ()
#12 0x370315ac in IODispatchCalloutFromCFMessage ()
#13 0x3866888a in __CFMachPortPerform ()
#14 0x386733e6 in __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ ()
#15 0x3867338a in __CFRunLoopDoSource1 ()
#16 0x3867220e in __CFRunLoopRun ()
#17 0x385e523c in CFRunLoopRunSpecific ()
#18 0x385e50c8 in CFRunLoopRunInMode ()
#19 0x3591e33a in GSEventRunModal ()
#20 0x379d5290 in UIApplicationMain ()
#21 0x0005aff4 in main at /Users/Sydin/App/main.m:16

【问题讨论】:

【参考方案1】:

在添加新注释时更改可见注释似乎存在某种问题——可能是MKMapView 中的问题,因为它可能在辅助线程中进行更新(就像那样挑剔)。我要做的是响应以下两个MKMapViewDelegate 方法:

mapView:regionWillChangeAnimated: mapView:regionDidChangeAnimated:

首先,设置一个标志,将地图指定为由用户移动到YES。在第二个中,将标志设置为NO。然后,每当您想添加新注释时,请检查该标志。如果是NO,那么您可以立即更新。如果是YES,请将您的注释添加到可变集合中。在mapView:regionDidChangeAnimated: 中,将该集合中的所有注释添加到地图视图中,然后再清空集合。

这应该可以解决问题,即使它不是一个优雅的解决方案。您可以在 Xcode 中打开断点窗格并单击添加按钮,然后选择“异常断点”,这样当您的应用程序崩溃时,您会得到准确的行而不是 main.m

【讨论】:

感谢您的建议;这确实成功了。另外,我应该更明确一点,但我在测试中确实使用了异常断点,这就是为什么我对没有获得有关异常的更多信息感到困惑的原因。【参考方案2】:

我有一个类似的问题,我正在添加一组从我的服务器获取的注释并将它们从 json 转换为 nsarray 但是因为这是在后台线程中完成的,当添加和删除到 mapview 我有 NSGenericException 时,我的解决方案是在主线程上添加删除操作,如下所示,一切正常,即使用户放大,您也可以添加删除注释。但是添加时会有轻微的延迟,因为我正在尝试 1k 到 2k 大小的数组。对于较小的东西应该没问题。

 dispatch_async(dispatch_get_main_queue(),
 ^
        [self.mapView removeAnnotations:self.mapView.annotations];
        [self.mapView addAnnotations:tempNSArray];

    );

【讨论】:

以上是关于如果在添加注释时缩放,MKMapView 会崩溃的主要内容,如果未能解决你的问题,请参考以下文章

相对于缩放级别缩放 MKMapView 注释

使用缩放后 MKMapView 应用程序崩溃

MKMapView 缩放以显示框架中的所有注释

MKMapView 不移动和缩放到搜索结果

单击注释气泡时应用程序崩溃

MKMapView 注释取消选择策略