1

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]];

上で行っていることは、MKMapView の可視領域内およびその周囲に (十分に) ランダムに分布する 100 個のポイントをマップに追加することです。これを再現する最も簡単な方法は、このコードをタイマー (たとえば 5 秒後) で実行するように設定し、マップを取得して、タイマーが実行されるまでズームインとズームアウトを少し開始することです。一度感じたら、毎回クラッシュする可能性があります。mapViewOutlet通話前にユーザーの操作を無効にaddAnnotationsしても、役に立たないようです。(おそらく、ユーザーがジェスチャーの途中である間、ユーザーの操作を無効にすることはできません。これは理解できると思います。)

私のもう 1 つの問題は、クラッシュをトレースする運があまりないことです。自分のコードで直接発生していないクラッシュを引き起こした経験があまりないため、トレースする明らかな方法を見逃している可能性があります。ですが、現在、 のreturn UIApplicationMain行で役に立たないブレークポイントを取得していますmain.m。私の推測では、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
4

2 に答える 2

4

新しいものを追加している間に、表示されている注釈への変更に何らかの問題があるように見えます — おそらくMKMapView、セカンダリ スレッドで更新を行っているため、おそらく の問題です (そのように厄介です)。私がすることは、次の2つのMKMapViewDelegate方法に応答することです。

  • mapView:regionWillChangeAnimated:
  • mapView:regionDidChangeAnimated:

最初に、マップがユーザーによって移動されたことを示すフラグを に設定しますYES。2 番目に、フラグを に設定しNOます。次に、新しい注釈を追加するたびに、フラグを確認します。の場合はNO、すぐに更新しても問題ありません。の場合YESは、注釈を変更可能なセットに追加します。で、セットを空にする前に、そのmapView:regionDidChangeAnimated:セット内の注釈をマップ ビューに追加します。

洗練された解決策でなくても、これで問題は解決するはずです。Xcode でブレークポイント ペインを開き、追加ボタンをクリックしてから、[例外ブレークポイント] を選択すると、アプリがクラッシュしたときに、main.m.

于 2012-12-18T00:17:24.887 に答える
1

サーバーからフェッチする注釈のグループを追加し、それらをjsonからnsarrayに変換する同様の問題がありましたが、これはマップビューへの追加および削除時にバックグラウンドスレッドで行われたため、 NSGenericException があり、私のソリューションは追加削除操作を実行していました以下のようなメインスレッドで、すべてが機能しています。ユーザーがズームアウトしているときでも、注釈を削除できます。ただし、1kから2kサイズの配列のようにしようとしているため、追加中にわずかな遅延が発生します。小さいものには問題ありません。

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

    });
于 2013-10-22T06:35:26.887 に答える