在回答您的问题时,您一次只能发送一个地理编码请求是正确的。事实上,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];
或者,您也可以使用更传统的模式。例如,您可以编写一个执行单个地理编码请求的方法,并在完成块中启动下一个请求,然后重复该过程,直到发出所有请求。