6

我正在尝试在 中显示多个位置MKMapItem。我从 a 获取这些位置CLGeocoder,不幸的是它只接受一个位置。即使我传入一个NSArray它只返回一个位置。

以下适用于单个位置,但不适用于多个位置。如何对多个位置进行地理编码?

Class mapItemClass = [MKMapItem class];
if (mapItemClass && [mapItemClass respondsToSelector:@selector(openMapsWithItems:launchOptions:)]) {
    NSArray *addresses = @[@"Mumbai",@"Delhi","Banglore"];

    CLGeocoder *geocoder = [[CLGeocoder alloc] init];
    [geocoder geocodeAddressString:@[addresses] completionHandler:^(NSArray *placemarks, NSError *error) {
        CLPlacemark *geocodedPlacemark = [placemarks objectAtIndex:0];
        MKPlacemark *placemark = [[MKPlacemark alloc] initWithCoordinate:geocodedPlacemark.location.coordinate addressDictionary:geocodedPlacemark.addressDictionary];
        MKMapItem *mapItem = [[MKMapItem alloc] initWithPlacemark:placemark];
        [mapItem setName:geocodedPlacemark.name];

        [MKMapItem openMapsWithItems:@[mapItem] launchOptions:nil];
    }];
}
4

2 回答 2

13

在回答您的问题时,您一次只能发送一个地理编码请求是正确的。事实上,CLGeocoder 类参考说我们的应用程序应该“为任何一个用户操作发送最多一个地理编码请求”。

因此,为此,您必须发送单独的请求。但是这些请求(异步运行)不应该同时运行。所以,问题是如何让一系列异步地理编码请求一个接一个地依次运行。

有很多不同的方法可以解决这个问题,但一种特别优雅的方法是使用并发子类,它在调用地理编码请求的异步完成块之前NSOperation不会完成操作(即不执行 KVN)。isFinished(有关并发操作的信息,请参阅并发编程指南的操作队列一章的为并发执行配置操作部分)。然后只需将这些操作添加到串行操作队列中。

另一种方法是使此异步地理编码请求以同步方式运行,然后您只需将请求添加到串行队列,请求将按顺序执行而不是并行执行。您可以通过使用信号量来实现这一点,有效地指示分派的任务在地理编码请求完成之前不返回。你可以这样做:

CLGeocoder *geocoder = [[CLGeocoder alloc]init];
NSMutableArray *mapItems = [NSMutableArray array];

NSOperationQueue *queue = [[NSOperationQueue alloc] init];
queue.maxConcurrentOperationCount = 1;   // make it a serial queue

NSOperation *completionOperation = [NSBlockOperation blockOperationWithBlock:^{
    [MKMapItem openMapsWithItems:mapItems launchOptions:nil];
}];

NSArray *addresses = @[@"Mumbai, India", @"Delhi, India", @"Bangalore, India"];

for (NSString *address in addresses) {
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);

        [geocoder geocodeAddressString:address completionHandler:^(NSArray *placemarks, NSError *error) {
            if (error) {
                NSLog(@"%@", error);
            } else if ([placemarks count] > 0) {
                CLPlacemark *geocodedPlacemark = [placemarks objectAtIndex:0];
                MKPlacemark *placemark = [[MKPlacemark alloc] initWithCoordinate:geocodedPlacemark.location.coordinate
                                                               addressDictionary:geocodedPlacemark.addressDictionary];
                MKMapItem *mapItem = [[MKMapItem alloc] initWithPlacemark:placemark];
                [mapItem setName:geocodedPlacemark.name];

                [mapItems addObject:mapItem];
            }
            dispatch_semaphore_signal(semaphore);
        }];

        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    }];

    [completionOperation addDependency:operation];
    [queue addOperation:operation];
}

[[NSOperationQueue mainQueue] addOperation:completionOperation];

或者,您也可以使用更传统的模式。例如,您可以编写一个执行单个地理编码请求的方法,并在完成块中启动下一个请求,然后重复该过程,直到发出所有请求。

于 2013-01-07T15:08:57.557 回答
1

对于寻找 Swift 解决方案的人:

func getCoordinate( addressString : String, completionHandler: @escaping(CLLocationCoordinate2D, NSError?) -> Void ){
            let geocoder = CLGeocoder()
            geocoder.geocodeAddressString(addressString) { (placemarks, error) in
                if error == nil {
                    if let placemark = placemarks?[0] {
                        let location = placemark.location!
                        completionHandler(location.coordinate, nil)
                        return
                    }
                }

                completionHandler(kCLLocationCoordinate2DInvalid, error as NSError?)
            }
        }

感谢苹果。官方文档链接

您可以通过以下方式将代码段用于多个地址:

let address = ["India","Nepal"]

for i in 0..<address.count {
 getCoordinate(addressString: address[i], completionHandler: { (location, error) in
                //Do your stuff here. For e.g. Adding annotation or storing location
      })

}
于 2017-12-20T15:52:32.473 回答