これは私のコード......
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
location_updated = [locations lastObject];
NSLog(@"updated coordinate are %@",location_updated);
latitude1 = location_updated.coordinate.latitude;
longitude1 = location_updated.coordinate.longitude;
self.lblLat.text = [NSString stringWithFormat:@"%f",latitude1];
self.lblLon.text = [NSString stringWithFormat:@"%f",longitude1];
NSString *str = [NSString stringWithFormat:@"https://maps.googleapis.com/maps/api/geocode/json?latlng=%f,%f&sensor=false",latitude1,longitude1];
url = [NSURL URLWithString:str];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
connection = [NSURLConnection connectionWithRequest:request delegate:self];
if (connection)
{
webData1 = [[NSMutableData alloc]init];
}
GMSMarker *marker = [[GMSMarker alloc] init];
marker.position = CLLocationCoordinate2DMake(latitude1,longitude1);
marker.title = formattedAddress;
marker.icon = [UIImage imageNamed:@"m2.png"];
marker.map = mapView_;
marker.draggable = YES;
}
このメソッドは複数回呼び出されますが、必要ありません。
そこにいくつかの制限を追加します。場所と精度の間のタイムスパン
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
CLLocation *newLocation = locations.lastObject;
NSTimeInterval locationAge = -[newLocation.timestamp timeIntervalSinceNow];
if (locationAge > 5.0) return;
if (newLocation.horizontalAccuracy < 0) return;
// Needed to filter cached and too old locations
//NSLog(@"Location updated to = %@", newLocation);
CLLocation *loc1 = [[CLLocation alloc] initWithLatitude:_currentLocation.coordinate.latitude longitude:_currentLocation.coordinate.longitude];
CLLocation *loc2 = [[CLLocation alloc] initWithLatitude:newLocation.coordinate.latitude longitude:newLocation.coordinate.longitude];
double distance = [loc1 distanceFromLocation:loc2];
if(distance > 20)
{
_currentLocation = newLocation;
//significant location update
}
//location updated
}
LocationManager
オブジェクトの割り当て中に、distanceFilter
のLocationManager
プロパティを設定できます。距離フィルタープロパティはCLLocationDistance
値であり、メートル単位で移動した距離をロケーションマネージャーに通知するように設定できます。次のように距離フィルターを設定できます。
LocationManager *locationManger = [[CLLocationManager alloc] init];
locationManager.delegate = self;
locationManager.distanceFilter = 100.0; // Will notify the LocationManager every 100 meters
locationManager.desiredAccuracy = kCLLocationAccuracyBest;
最も簡単な方法:
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
{
[manager stopUpdatingLocation];
manager.delegate = nil;
//...... do something
}
マネージャーはdidUpdateLocationsメソッドをdelegate参照なしで見つけることができません:-D
ただし、startUpdatingLocationを使用する前に、再度設定することを忘れないでください
同様の状況があります。 dispatch_onceを使用できます:
static dispatch_once_t predicate;
- (void)update
{
if ([CLLocationManager authorizationStatus] == kCLAuthorizationStatusNotDetermined &&
[_locationManager respondsToSelector:@selector(requestWhenInUseAuthorization)]) {
[_locationManager requestWhenInUseAuthorization];
}
_locationManager.delegate = self;
_locationManager.distanceFilter = kCLDistanceFilterNone;
_locationManager.desiredAccuracy = kCLLocationAccuracyBest;
predicate = 0;
[_locationManager startUpdatingLocation];
}
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations {
[manager stopUpdatingLocation];
manager = nil;
dispatch_once(&predicate, ^{
//your code here
});
}
次のように、静的変数を使用して最新の場所のタイムスタンプを保存し、それを最新のものと比較できます。
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
[manager stopUpdatingLocation];
static NSDate *previousLocationTimestamp;
CLLocation *location = [locations lastObject];
if (previousLocationTimestamp && [location.timestamp timeIntervalSinceDate:previousLocationTimestamp] < 2.0) {
NSLog(@"didUpdateLocations GIVE UP");
return;
}
previousLocationTimestamp = location.timestamp;
NSLog(@"didUpdateLocations GOOD");
// Do your code here
}
次のように書くことができます:[manager stopUpdatingLocation]; manager = nil; didupdatelocationデリゲートで
時間の制約については、受け入れられた回答からコードを理解できず、別のアプローチを投稿しました。ロブが「初めて位置情報サービスを開始すると、複数回呼び出されることがあります」と指摘しています。以下のコードは最初の場所で動作し、最初の120秒間は更新された場所を無視します。元の質問「didUpdateLocationsのメソッド呼び出しを複数回停止する方法」に対処する1つの方法です。
.hファイル内:
@property(strong,nonatomic) CLLocation* firstLocation;
.mファイル内:
// is this the first location?
CLLocation* newLocation = locations.lastObject;
if (self.firstLocation) {
// app already has a location
NSTimeInterval locationAge = [newLocation.timestamp timeIntervalSinceDate:self.firstLocation.timestamp];
NSLog(@"locationAge: %f",locationAge);
if (locationAge < 120.0) { // 120 is in seconds or milliseconds?
return;
}
} else {
self.firstLocation = newLocation;
}
// do something with location
ロケーションマネージャーの更新を停止したいときにこのメソッドを記述します
[locationManager stopUpdatingLocation];
フラグ(Bool)を設定できます。 locationsManagerをインスタンス化すると、flag = trueに設定され、その後locationManager:didUpdateLocationsが、フラグ= falseに設定された場合にのみ実行するコードブロック内に戻る場合。この方法では、一度だけ実行されます。
if flag == true {
flag = false
...some code probably network call you only want to run the once
}
ロケーションマネージャーは複数回呼び出されますが、実行したいコードは1回だけです。
locationManager.startUpdatingLocation()はロケーションを連続的にフェッチし、didUpdateLocationsメソッドは数回呼び出します。locationManager.startUpdatingLocation()を呼び出す前に、locationManager.distanceFilter値の値を設定するだけです。
200メートルを設定すると(要件に応じて変更できます)正常に動作します
locationManager = CLLocationManager()
locationManager.delegate = self
locationManager.desiredAccuracy = kCLLocationAccuracyBest
locationManager.distanceFilter = 200
locationManager.requestWhenInUseAuthorization()
locationManager.startUpdatingLocation()