IOS之定位淺談

由于項目的需要,用到定位的功能,折騰了一下午,總算有個大概得了解了。

這里主要是用了CLLocationManager進行定位。使用分為以下幾個步驟:

1.導入相關的庫CoreLocation.frameWork

2.勾選開啟后臺定位選項

3.此時在plist文件可以看到多了

4.在plist添加兩個參數,允許應用在使用期間定位、或者在后臺也能繼續定位。


5.到這里為止,所有的界面配置完成。然后在AppDelegate.h類中導入頭文件

```

#import<CoreLocation/CoreLocation.h>

```

若是報錯,則證明在第一步中沒有導入相關的 CoreLocation.frameWork 庫。

在AppDelegate.m文件中有代理方法

```

@interface AppDelegate ()<CLLocationManagerDelegate>

```

然后在APPDelegate.h中聲明設置這些變量。

```

@property (strong,nonatomic)CLLocationManager * locationManager; // 創建工程的全部變量

@property (nonatomic,assign)BOOL executingInBackground;

@property (strong,nonatomic)NSTimer * timer; // 計時器

```

一般來說,locationManager定位只要一開始,如果不[self.locationManager stopUpdatingLocation],則會根據所在的位置是否超出多少米,不定時的更新定位。如果想要每隔一段時間進行定位。則需要開啟一個計時器(timer),然后每定位一次,立刻關掉定位。待計時器計算時間,再次觸發定位。在didFinishLaunchingWithOptions方法中:

```

self.locationManager = [[CLLocationManager alloc]init];

self.locationManager.delegate = self;

//? ? self.locationManager.distanceFilter = 1.0; // 位置超過1米,就再次重新定位

self.locationManager.desiredAccuracy = kCLLocationAccuracyBest; // 定位精度最為準確,越準確,耗電量越大

//? ? [self.locationManager startUpdatingLocation];

self.locationManager.pausesLocationUpdatesAutomatically=NO;

if([self.locationManager respondsToSelector:@selector(requestAlwaysAuthorization)])

{

if(kIOSVersions>=8.0)

{

[self.locationManager requestAlwaysAuthorization];

[self.locationManager requestWhenInUseAuthorization];//使用程序其間允許訪問位置數據(iOS8定位需要)

}

if(kIOSVersions>=9.0)

{

[self.locationManager setAllowsBackgroundLocationUpdates:YES];

}

}


NSError *setCategoryErr = nil;

NSError *activationErr? = nil;

[[AVAudioSession sharedInstance]

setCategory: AVAudioSessionCategoryPlayback

error: &setCategoryErr];

[[AVAudioSession sharedInstance]

setActive: YES

error: &activationErr];

NSUserDefaults *userDefault = [NSUserDefaults standardUserDefaults];

[userDefault setValue:@"0" forKey:isShowTestEnvironment];

[userDefault setBool:NO forKey:USERDEFAULTS_IS_SAVE_ENVIRONMENT];

[userDefault synchronize];

```

以上代碼之所以沒有立刻開啟定位,是想讓計時器觸發開始定位。

6.應用程序掉到后臺時候,執行的方法:applicationDidEnterBackground

要想在后臺計時器能繼續計時,則需要在此方法中添加

```

UIApplication*? app = [UIApplication sharedApplication];

__block? ? UIBackgroundTaskIdentifier bgTask;

bgTask = [app beginBackgroundTaskWithExpirationHandler:^{

dispatch_async(dispatch_get_main_queue(), ^{

if (bgTask != UIBackgroundTaskInvalid)

{

bgTask = UIBackgroundTaskInvalid;

}

});

}];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

dispatch_async(dispatch_get_main_queue(), ^{

if (bgTask != UIBackgroundTaskInvalid)

{

bgTask = UIBackgroundTaskInvalid;

}

});

});

```

7.應用在前臺是調用的方法:applicationWillEnterForeground

//當應用程序回到前臺時,執行該方法

```

-(void)applicationWillEnterForeground:(UIApplication*)application

{

//程序進入前臺,轉化為高精確定位

self.executingInBackground = NO;

[self.locationManager setDesiredAccuracy:kCLLocationAccuracyBest];

}

```

8.定位成功時候的回調方法:didUpdateToLocation

```

#pragma mark 定位成功回調

- (void)locationManager:(CLLocationManager *)manager didUpdateToLocation:(CLLocation *)newLocation fromLocation:(CLLocation *)oldLocation

{

if (self.executingInBackground)

{ // 在后臺

if(kIOSVersions>=8.0)

{

[self.locationManager requestAlwaysAuthorization];

[self.locationManager requestWhenInUseAuthorization];//使用程序其間允許訪問位置數據(iOS8定位需要)

}

if(kIOSVersions>=9.0)

{

[self.locationManager setAllowsBackgroundLocationUpdates:YES];

}

NSLogS(@"后臺臺");

}else{

// 在前臺

NSLogS(@"前臺");

}

NSLog(@"經度:%f", newLocation.coordinate.longitude+0.001253);

NSLog(@"緯度:%f", newLocation.coordinate.latitude-0.000182);

NSLog(@"速度:%f 米/秒", newLocation.speed);

CLGeocoder * geocoder = [[CLGeocoder alloc] init];

[geocoder reverseGeocodeLocation:newLocation completionHandler:^(NSArray *placemarks, NSError *error) {

NSDictionary *locationInfo = [[NSDictionary alloc]init];

for (CLPlacemark * placemark in placemarks) {

locationInfo = [placemark addressDictionary];

}

NSLog(@"%@",locationInfo);

}];

```

[self.locationManager stopUpdatingLocation]; // 這里每定位一次,就關掉定位,等待計時器再次觸發。如果不關掉定位,則:

?? self.locationManager.distanceFilter = 1.0; // 兩者位置超過1米,就再次重新定位

根據這句話判斷是否重新定位。(需不需要關掉,看需求)

}

9.定位失敗的時候的回調方法

```

#pragma mark 定位失敗回調

- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error

{

NSLog(@"error%@", error);

}

```

ps:這里我觸發定位的開始是在MainMenuController類中做的。(也就是一旦用戶登錄成功就開始定位)

在MainMenuController類中:

調用計時器的圖片

說得比較粗糙,望各位多多指點,謝謝~

ps:在ios8系統中,不知怎么的,我退到后臺定位后,利用計時器調用觸發事件,只執行了兩次就沒法點位了,后來檢查了好久,發現計時器在后臺還一直計時,被困擾了好久。最后,刪了app,重新再安裝,就可以了。真暈,不知道是不是緩存的問題。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容